
Java 教程
- Java - 首頁
- Java - 概述
- Java - 歷史
- Java - 特性
- Java 與 C++
- JVM - Java 虛擬機器
- Java - JDK 與 JRE 與 JVM
- Java - Hello World 程式
- Java - 環境設定
- Java - 基本語法
- Java - 變數型別
- Java - 資料型別
- Java - 型別轉換
- Java - Unicode 系統
- Java - 基本運算子
- Java - 註釋
- Java - 使用者輸入
- Java - 日期和時間
Java 控制語句
- Java - 迴圈控制
- Java - 決策制定
- Java - If-else
- Java - Switch
- Java - For 迴圈
- Java - For-Each 迴圈
- Java - While 迴圈
- Java - do-while 迴圈
- Java - Break
- Java - Continue
面向物件程式設計
- Java - OOPs 概念
- Java - 物件和類
- Java - 類屬性
- Java - 類方法
- Java - 方法
- Java - 變數作用域
- Java - 建構函式
- Java - 訪問修飾符
- Java - 繼承
- Java - 聚合
- Java - 多型
- Java - 重寫
- Java - 方法過載
- Java - 動態繫結
- Java - 靜態繫結
- Java - 例項初始化塊
- Java - 抽象
- Java - 封裝
- Java - 介面
- Java - 包
- Java - 內部類
- Java - 靜態類
- Java - 匿名類
- Java - 單例類
- Java - 包裝類
- Java - 列舉
- Java - 列舉建構函式
- Java - 列舉字串
Java 內建類
Java 檔案處理
Java 錯誤和異常
- Java - 異常
- Java - try-catch 塊
- Java - try-with-resources
- Java - 多重 catch 塊
- Java - 巢狀 try 塊
- Java - Finally 塊
- Java - throw 異常
- Java - 異常傳播
- Java - 內建異常
- Java - 自定義異常
Java 多執行緒
- Java - 多執行緒
- Java - 執行緒生命週期
- Java - 建立執行緒
- Java - 啟動執行緒
- Java - 執行緒連線
- Java - 執行緒命名
- Java - 執行緒排程器
- Java - 執行緒池
- Java - 主執行緒
- Java - 執行緒優先順序
- Java - 守護執行緒
- Java - 執行緒組
- Java - 關閉鉤子
Java 同步
Java 網路
- Java - 網路
- Java - 套接字程式設計
- Java - URL 處理
- Java - URL 類
- Java - URLConnection 類
- Java - HttpURLConnection 類
- Java - Socket 類
- Java - 泛型
Java 集合
Java 介面
Java 資料結構
Java 集合演算法
高階 Java
- Java - 命令列引數
- Java - Lambda 表示式
- Java - 傳送電子郵件
- Java - Applet 基礎
- Java - Javadoc 註釋
- Java - 自動裝箱和拆箱
- Java - 檔案不匹配方法
- Java - REPL (JShell)
- Java - 多版本 Jar 檔案
- Java - 私有介面方法
- Java - 內部類菱形運算子
- Java - 多解析度影像 API
- Java - 集合工廠方法
- Java - 模組系統
- Java - Nashorn JavaScript
- Java - Optional 類
- Java - 方法引用
- Java - 函式式介面
- Java - 預設方法
- Java - Base64 編碼解碼
- Java - Switch 表示式
- Java - Teeing 收集器
- Java - 微基準測試
- Java - 文字塊
- Java - 動態 CDS 存檔
- Java - Z 垃圾收集器 (ZGC)
- Java - 空指標異常
- Java - 打包工具
- Java - 密封類
- Java - 記錄類
- Java - 隱藏類
- Java - 模式匹配
- Java - 緊湊數字格式化
- Java - 垃圾回收
- Java - JIT 編譯器
Java 雜項
- Java - 遞迴
- Java - 正則表示式
- Java - 序列化
- Java - 字串
- Java - 程序 API 改進
- Java - 流 API 改進
- Java - 增強的 @Deprecated 註釋
- Java - CompletableFuture API 改進
- Java - 流
- Java - 日期時間 API
- Java 8 - 新特性
- Java 9 - 新特性
- Java 10 - 新特性
- Java 11 - 新特性
- Java 12 - 新特性
- Java 13 - 新特性
- Java 14 - 新特性
- Java 15 - 新特性
- Java 16 - 新特性
Java API 和框架
Java 類引用
- Java - Scanner
- Java - 陣列
- Java - 字串
- Java - Date
- Java - ArrayList
- Java - Vector
- Java - Stack
- Java - PriorityQueue
- Java - LinkedList
- Java - ArrayDeque
- Java - HashMap
- Java - LinkedHashMap
- Java - WeakHashMap
- Java - EnumMap
- Java - TreeMap
- Java - IdentityHashMap
- Java - HashSet
- Java - EnumSet
- Java - LinkedHashSet
- Java - TreeSet
- Java - BitSet
- Java - Dictionary
- Java - Hashtable
- Java - Properties
- Java - Collection
- Java - Array
Java 有用資源
Java - 隱藏類
Java 15 引入了**隱藏類**,其他類的位元組碼無法直接使用這些類。這些**隱藏類**旨在供在執行時生成類的框架使用,並使用反射來使用它們。
**隱藏類**定義為基於訪問控制上下文的成員,並且可以獨立於其他類解除安裝。
此提案 (JEP 371) 旨在透過提供一個標準 API 來定義不可發現且生命週期有限的**隱藏類**,從而改進 JVM 上的所有語言。JDK 框架或外部框架可以動態生成類,這些類可以生成隱藏類。
JVM 語言嚴重依賴於動態類生成以實現靈活性和效率。
目標
以下是此增強功能的目標列表
- 框架應該能夠將類定義為框架的不可發現的實現細節,這些類既不能連結到其他類,也不能使用反射發現。
- 使用不可發現的類擴充套件訪問控制巢狀。
- 積極解除安裝隱藏類將幫助框架根據需要定義儘可能多的隱藏類,而不會降低效能。
- 棄用非標準 API,misc.Unsafe::defineAnonymousClass,以便在將來的版本中刪除。
建立隱藏類
為了建立隱藏類,我們必須建立一個 Lookup 例項,如下所示
MethodHandles.Lookup lookup = MethodHandles.lookup();
一旦 Lookup 例項可用,我們就可以使用 defineHiddenClass() 方法使用隱藏類的位元組陣列來建立隱藏類。
Class<?> hiddenClass = lookup.defineHiddenClass(getByteArray(), true, ClassOption.NESTMATE).lookupClass();
隱藏類的位元組陣列可以使用隱藏類的類路徑檢索。
public static byte[] getByteArray() throws IOException { InputStream stream = Util.class.getClassLoader().getResourceAsStream("com/tutorialspoint/Util.class"); byte[] bytes = stream.readAllBytes(); return bytes; }
一旦載入了 hiddenClass 類,我們就可以使用 getConstructor() 方法建立其例項。
Object hiddenClassObj = hiddenClass.getConstructor().newInstance();
使用隱藏類例項,我們可以獲取方法,然後像下面這樣執行它
Method method = hiddenClassObj.getClass().getDeclaredMethod("square", Integer.class); // call the method and get result Object result = method.invoke(hiddenClassObj, 3);
由於隱藏類是隱藏的,並且不能使用反射例項化,因此其隱藏屬性為 true,規範名稱為 null。
建立和使用隱藏類的示例
以下示例顯示了隱藏類的建立和使用。首先,我們建立了一個公共類 Util,如下所示
package com.tutorialspoint; public class Util { public Integer square(Integer n) { return n * n; } }
現在,我們已將此類建立為隱藏類,並訪問其方法以獲取數字的平方。
package com.tutorialspoint; import java.io.IOException; import java.io.InputStream; import java.lang.invoke.MethodHandles; import java.lang.invoke.MethodHandles.Lookup.ClassOption; import java.lang.reflect.InvocationTargetException; import java.lang.reflect.Method; public class Tester { public static void main(String args[]) throws IllegalAccessException, IOException, InstantiationException, IllegalArgumentException, InvocationTargetException, NoSuchMethodException, SecurityException { // create the lookup object MethodHandles.Lookup lookup = MethodHandles.lookup(); // define the hidden class using the byte array of Util class // Using NESTMATE option so that hidden class has access to // private members of classes in same nest Class<?> hiddenClass = lookup.defineHiddenClass(getByteArray(), true, ClassOption.NESTMATE).lookupClass(); // get the hidden class object Object hiddenClassObj = hiddenClass.getConstructor().newInstance(); // get the hidden class method Method method = hiddenClassObj.getClass().getDeclaredMethod("square", Integer.class); // call the method and get result Object result = method.invoke(hiddenClassObj, 3); // print the result System.out.println(result); // as hidden class is not visible to jvm, it will print hidden System.out.println(hiddenClass.isHidden()); // canonical name is null thus this class cannot be instantiated using reflection System.out.println(hiddenClass.getCanonicalName()); } public static byte[] getByteArray() throws IOException { InputStream stream = Util.class.getClassLoader().getResourceAsStream("com/tutorialspoint/Util.class"); byte[] bytes = stream.readAllBytes(); return bytes; } }
讓我們編譯並執行以上程式,這將產生以下結果 -
9 true null
廣告