第七讲 HiddenClass
视频课:https://edu.csdn.net/lecturer/1516
JEP 371:Hidden Classes(隐藏类)
该提案通过启用标准 API 来定义无法发现且具有有限生命周期的隐藏类,从而提高 JVM 上所有语言的效率。JDK内部和外部的框架将能够动态生成类,而这些类可以定义隐藏类。通常来说基于JVM的很多语言都有动态生成类的机制,这样可以提高语言的灵活性和效率。
- 隐藏类天生为框架设计的,在运行时生成内部的class。
- 隐藏类只能通过反射访问,不能直接被其他类的字节码访问。
- 隐藏类可以独立于其他类加载、卸载,这可以减少框架的内存占用。
一.概念
不能直接被其他class的二进制代码使用的class。Hidden Classes主要被一些框架用来生成运行时类,但是这些类不是被用来直接使用的,而是通过反射机制来调用。
比如在JDK8中引入的lambda表达式,JVM并不会在编译的时候将lambda表达式转换成为专门的类,而是在运行时将相应的字节码动态生成相应的类对象。
二.为什么要用
如果标准API可以定义隐藏的不可发现且生命周期有限的类,那么动态生成类的JDK内外的框架可以定义隐藏类。JavaScript引擎可以为JavaScript程序翻译的字节码生成隐藏类,因为知道当引擎不再使用这些类时,这些类将被卸载。灵活性、安全性,保证这两点,框架基础实现。
举栗子:
public class HiddenTest {
public static void main(String[] args) {
Runnable runnable=()->{
System.out.println("JDK15测试");
};
}
}
反编译:
public class HiddenTest {
public HiddenTest() {
}
public static void main(String[] args) {
Runnable runnable = () -> {
System.out.println("JDK15测试");
};
}
}
JClassLib截图:
[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-krbIuZMS-1607086350659)(E:\Java15新特性\assets\image-20201201232359801.png)]
invokedynamic:运行时动态解析调用的类;
invokevirtual:调用实例方法
三.特性和支持
3.1 动态生成类特性
- 不可发现性。因为我们是为某些静态的类动态生成的动态类,所以我们希望把这个动态生成的类看做是静态类的一部分。所以我们不希望除了该静态类之外的其他机制发现。
- 访问控制。我们希望在访问控制静态类的同时,也能控制到动态生成的类。
- 生命周期。动态生成类的生命周期一般都比较短,我们并不需要将其保存和静态类的生命周期一致。
API的支持
所以我们需要一些API来定义无法发现的且具有有限生命周期的隐藏类。这将提高所有基于JVM的语言实现的效率。
比如:
java.lang.reflect.Proxy可以定义隐藏类作为实现代理接口的代理类。
java.lang.invoke.StringConcatFactory可以生成隐藏类来保存常量连接方法;
java.lang.invoke.LambdaMetaFactory可以生成隐藏的nestmate类,以容纳访问封闭变量的lambda主体;
java.lang.invoke.LambdaMetaFactory可以生成隐藏的nestmate类,以容纳访问封闭变量的lambda主体;
普通类是通过调用ClassLoader::defineClass创建的,而隐藏类是通过调用Lookup::defineHiddenClass创建的。这使JVM从提供的字节中派生一个隐藏类,链接该隐藏类,并返回提供对隐藏类的反射访问的查找对象。调用程序可以通过返回的查找对象来获取隐藏类的Class对象。