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
广告