Java 15 新特性—Hidden Classes(隐藏类)

作者简介:大家好,我是smart哥,前中兴通讯、美团架构师,现某互联网公司CTO

联系qq:184480602,加我进群,大家一起学习,一起进步,一起对抗互联网寒冬

学习必须往深处挖,挖的越深,基础越扎实!

阶段1、深入多线程

阶段2、深入多线程设计模式

阶段3、深入juc源码解析


阶段4、深入jdk其余源码解析


阶段5、深入jvm源码解析

码哥源码部分

码哥讲源码-原理源码篇【2024年最新大厂关于线程池使用的场景题】

码哥讲源码【炸雷啦!炸雷啦!黄光头他终于跑路啦!】

码哥讲源码-【jvm课程前置知识及c/c++调试环境搭建】

​​​​​​码哥讲源码-原理源码篇【揭秘join方法的唤醒本质上决定于jvm的底层析构函数】

码哥源码-原理源码篇【Doug Lea为什么要将成员变量赋值给局部变量后再操作?】

码哥讲源码【你水不是你的错,但是你胡说八道就是你不对了!】

码哥讲源码【谁再说Spring不支持多线程事务,你给我抽他!】

终结B站没人能讲清楚红黑树的历史,不服等你来踢馆!

打脸系列【020-3小时讲解MESI协议和volatile之间的关系,那些将x86下的验证结果当作最终结果的水货们请闭嘴】

什么是隐藏类

隐藏类,是一种不能被其他类直接使用的类。Java 15 引入隐藏类主要针对的是库和框架的开发者,而不是直接面向普通 Java 应用程序开发者。它有如下几个特点:

  • 不可见性:隐藏类对于 Java 的反射 API 是不可见的,这意味着它们不能通过正常的反射机制被发现或访问。但是,这并不是说,他们是完全不可见的,我们需要知道访问他们的“密码”,只要知道这个密码就可以访问他们。隐藏类与普通 Java 类的最大区别就是隐藏类并不是“广而告之”的,需要通过特殊的手段来找到他们。
  • 不兼容性:隐藏类与普通的 Java 类不兼容,这意味着我们不能将一个隐藏类实例转换为任何非隐藏类,也不能将非隐藏类转换为隐藏类。
  • 生命周期管理:隐藏类的生命周期可以由创建它们的框架或库更精细地控制。当它们不再被需要时,可以被卸载,这有助于资源管理和性能优化。

怎么使用隐藏类

隐藏类不是通过常规的 Java 代码创建的,而是通过特定的 API 调用在运行时动态生成。下面大明哥来演示下如何使用隐藏类。

第一步:创建一个普通的 Java 类

public class HiddenClasses {

    public void print() {
        System.out.println("hello,sike java!!!");
    }
}

第二步:编译 Java 类

使用 javac 命令编译该文件,得到 .class 文件,如下:

javac HiddenClasses.java

第三步:读取字节码

一旦类被编译成 .class 文件,我们就可以通过读取这个文件来获取其字节码。

Path path = Paths.get("HiddenClasses.class");
// 字节码
byte[] bytes = Files.readAllBytes(path);
// 使用 base64 对其 Encode
System.out.println(Base64.getEncoder().encodeToString(bytes));

执行得到结果为:

yv66vgAAADQAHAoABgAOCQAPABAIABEKABIAEwcAFAcAFQEABjxpbml0PgEAAygpVgEABENvZGUBAA9MaW5lTnVtYmVyVGFibGUBAAVwcmludAEAClNvdXJjZUZpbGUBABJIaWRkZW5DbGFzc2VzLmphdmEMAAcACAcAFgwAFwAYAQAUaGVsbG/vvIxzaWtlIGphdmEhISEHABkMABoAGwEAJWNvbS9za2phdmEvamF2YS9mZWF0dXJlL0hpZGRlbkNsYXNzZXMBABBqYXZhL2xhbmcvT2JqZWN0AQAQamF2YS9sYW5nL1N5c3RlbQEAA291dAEAFUxqYXZhL2lvL1ByaW50U3RyZWFtOwEAE2phdmEvaW8vUHJpbnRTdHJlYW0BAAdwcmludGxuAQAVKExqYXZhL2xhbmcvU3RyaW5nOylWACEABQAGAAAAAAACAAEABwAIAAEACQAAAB0AAQABAAAABSq3AAGxAAAAAQAKAAAABgABAAAAAwABAAsACAABAAkAAAAlAAIAAQAAAAmyAAISA7YABLEAAAABAAoAAAAKAAIAAAAGAAgABwABAAwAAAACAA0=

第四步:创建隐藏类并调用其方法

public class Test {
    public static void main(String[] args) throws Throwable {
        // 创建一个 Lookup 实例
        MethodHandles.Lookup lookup = MethodHandles.lookup();

        // 将刚刚生成的 base64 解码
        byte[] classBytes = Base64.getDecoder().decode("yv66vgAAADQAHAoABgAOCQAPABAIABEKABIAEwcAFAcAFQEABjxpbml0PgEAAygpVgEABENvZGUBAA9MaW5lTnVtYmVyVGFibGUBAAVwcmludAEAClNvdXJjZUZpbGUBABJIaWRkZW5DbGFzc2VzLmphdmEMAAcACAcAFgwAFwAYAQAUaGVsbG/vvIxzaWtlIGphdmEhISEHABkMABoAGwEAJWNvbS9za2phdmEvamF2YS9mZWF0dXJlL0hpZGRlbkNsYXNzZXMBABBqYXZhL2xhbmcvT2JqZWN0AQAQamF2YS9sYW5nL1N5c3RlbQEAA291dAEAFUxqYXZhL2lvL1ByaW50U3RyZWFtOwEAE2phdmEvaW8vUHJpbnRTdHJlYW0BAAdwcmludGxuAQAVKExqYXZhL2xhbmcvU3RyaW5nOylWACEABQAGAAAAAAACAAEABwAIAAEACQAAAB0AAQABAAAABSq3AAGxAAAAAQAKAAAABgABAAAAAwABAAsACAABAAkAAAAlAAIAAQAAAAmyAAISA7YABLEAAAABAAoAAAAKAAIAAAAGAAgABwABAAwAAAACAA0=");

        // 定义隐藏类
        Class<?> hiddenClass = lookup.defineHiddenClass(classBytes, true, MethodHandles.Lookup.ClassOption.NESTMATE).lookupClass();

        // 创建隐藏类的实例
        Object hiddenClassInstance = hiddenClass.getConstructor().newInstance();

        // 获取 print 方法的 MethodHandle
        var printHandle = lookup.findVirtual(hiddenClass, "print", MethodType.methodType(void.class));

        // 调用 print 方法
        printHandle.invoke(hiddenClassInstance);
    }
}

执行得到结果:

整体就分为两个步骤:

  1. 生成字节码。可以使用 ASM 这样的类库来生成,也可以通过大明哥这种方式,这种方式的优点在于它非常直观。
  2. 反射调用方法。使用 Lookup 、 MethodHandles 和 MethodType 来创建隐藏类和调用方法。

我们需要注意的,隐藏类,是一个高级特性,主要用于库和框架的开发者,并不是面向普通的应用程序猿。而且,由于涉及到底层的字节码操作和类加载机制,所以在使用这一特性时,我们需要对 Java 的内部工作机制有深入的理解。

  • 18
    点赞
  • 7
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值