如何 0 成本启动全员 AI 技能提升?戳> 了解详情
写点什么

Java 架构师:ASM 已经过时,建议使用类文件 API 简化开发

  • 2024-01-07
    北京
  • 本文字数:2170 字

    阅读完需:约 7 分钟

大小:605.33K时长:03:26
Java架构师:ASM已经过时,建议使用类文件API简化开发

JEP 457,类文件API(Class-File API,预览),最近已经集成到了 JDK 22 中。该 JEP 建议提供一个 API 来解析、生成和转换 Java 类文件。它最初将作为 JDK 中 Java 字节码操作和分析框架ASM的内部替代品,并计划将其作为公共 API 开放。甲骨文的 Java 语言架构师Brian Goetz将 ASM 描述为“带有大量遗留包袱的旧代码库”,并提供了关于该草案将如何发展并最终取代 ASM 的背景信息

 

类文件 API 的核心是几项关键原则。首先,它将类文件实体(比如字段、方法、属性和字节码指令)均视为不可变对象。这种不可变的表述确保在类文件转换时可以可靠地进行共享。该 API 采用树形结构来反映类文件的层次结构,从而支持用户驱动的导航以进行高效地解析。它还强调了解析过程的延迟性,即只会处理满足用户需求所需的类文件。

 

类文件 API 位于java.lang.classfile包及其子包中,包含三个主要的抽象,即元素、构建器和转换。元素是类文件组件的不可变描述。构建器对应于每种复合元素,可以使用特定的构建方法来方便地构造类文件。转换表示在构建过程中修改元素的函数。

该 API 还引入了使用模式解析类文件的新方法,与 ASM 基于访问者(visitor)的方式有所不同。这支持更直接和更简洁的表达式,利用了 Java 的模式匹配功能。例如,开发人员可以遍历 CodeModel 中的指令,并匹配感兴趣的元素,以完成像依赖图构造这样的任务。

 

考虑如下的样例:

CodeModel code = ...;Set<ClassDesc> deps = new HashSet<>();for (CodeElement e : code) {    switch (e) {        case FieldInstruction f  -> deps.add(f.owner());        case InvokeInstruction i -> deps.add(i.owner());        // ... and so on for instanceof, cast, etc ...    }}
复制代码

这个代码片段展示了使用模式匹配来解析 Code 属性以收集类依赖关系图的依赖,迭代指令并匹配特定的类型。

 

使用构建器生成类文件是另一项关键特性。该 API 颠覆了使用构造函数或工厂创建构建器的传统习惯,相反,客户端提供一个接受构建器的 lambda。这种方法提供了更具体和透明的代码生成,并且可以重放操作序列。它还为管理块范围、局部变量索引计算和标签管理提供了更高级的便利性。

 

下面的代码展示了如何使用构建器生成方法,演示了该 API 具体且透明的代码生成方式。

ClassBuilder classBuilder = ...;classBuilder.withMethod("fooBar", MethodTypeDesc.of(CD_void, CD_boolean, CD_int), flags,    methodBuilder -> methodBuilder.withCode(codeBuilder -> {        Label label1 = codeBuilder.newLabel();        Label label2 = codeBuilder.newLabel();        codeBuilder.iload(1)            .ifeq(label1)            .aload(0)            .iload(2)            .invokevirtual(ClassDesc.of("Foo"), "foo", MethodTypeDesc.of(CD_void, CD_int))            .goto_(label2)            .labelBinding(label1)            .aload(0)            .iload(2)            .invokevirtual(ClassDesc.of("Foo"), "bar", MethodTypeDesc.of(CD_void, CD_int))            .labelBinding(label2)            .return_();    });
复制代码

类文件的转换功能同样值得注意。解析和生成方法保持了一致,这样转换可以无缝进行。例如,开发人员可以处理一个类,以便于删除特定的方法,或者通过应用各种转换来改变方法体。

 

下面的代码片段展示了 API 的转换类文件的功能,演示了在转换过程中如何有选择地修改或替换类元素。

ClassFile cf = ClassFile.of();ClassModel classModel = cf.parse(bytes);byte[] newBytes = cf.transform(classModel, (classBuilder, ce) -> {    if (ce instanceof MethodModel mm) {        classBuilder.transformMethod(mm, (methodBuilder, me)-> {            if (me instanceof CodeModel cm) {                methodBuilder.transformCode(cm, (codeBuilder, e) -> {                    switch (e) {                        case InvokeInstruction i                                when i.owner().asInternalName().equals("Foo") ->                            codeBuilder.invokeInstruction(i.opcode(), ClassDesc.of("Bar"),                                                           i.name().stringValue(),                                                          i.typeSymbol(), i.isInterface());                        default -> codeBuilder.with(e);                    }                });            }            else                methodBuilder.with(me);        });    }    else        classBuilder.with(ce);});
复制代码

JEP 457 具有变革性的一个方面是如何解决 Java 生态系统中类文件格式的快速发展所带来的挑战。通过提供与 JDK 一起演进的标准 API,它能够确保使用该 API 的框架和工具会自动支持来自最新 JDK 的类文件。这种能力对于新语言和 VM 特性非常重要,它们在类文件中可能会有相应地表述。

 

总之,JEP 457 的类文件 API 是一种具有前瞻性的解决方案,符合 Java 开发的现代化需求。它的设计原则、抽象和转换功能使其成为 Java 开发人员的强大工具,增强了 Java 生态系统中类文件管理的效率和可靠性。

 

原文链接:

 JEP 457: Streamlining Java Development with the Class-File API

2024-01-07 08:0010195

评论

发布
暂无评论
发现更多内容

下次面试再一上来就问我线程有哪些状态,我上去就是给他一 jio

钟奕礼

Java 程序员 java面试 java编程

一文了解 Go 接口

陈明勇

Go golang 接口

信号量的无序竞争和有序竞争

eng八戒

c Linux Kenel 信号量 C++

面对当下最热的多模态,为什么这些业界和学界专家说“不必追热点”

小红书技术REDtech

防治“虚假种草”,小红书技术团队干了这几件大事

小红书技术REDtech

“算法考核没过,老板找我聊了3个小时”:离职程序员重刷数学题

钟奕礼

Java 程序员 java面试 java编程

Python初学者必备!适合新手阅读的Github开源代码。。。

Jackpop

Linux Network RSS RPS 疑惑

Geek_f24c45

container network veth RPS

AngularJS进阶(十五)Cookie ‘data‘ possibly not set or overflowed because it was too large

No Silver Bullet

Cookie AngularJS 12月月更

8年程序员年初被迫毕业,前后面试30家公司,如今终于拿到Offer!

程序员小毕

程序员 面试 程序人生 后端 架构师

思路一转,春暖花开!动动手指,这段程序性能又双叒提升2s~15s

靠谱的程序员

MySQL MyBatisPlus Mybatis-Plus

迎战大厂!“金九银十”和秋招通过率达95%的Java面试要点集锦

钟奕礼

Java 程序员 java面试 java编程

33K Star!这才是程序员需要的神器。。。

Jackpop

MySQL锁,锁的到底是什么?

Java永远的神

MySQL 数据库 程序员 面试 后端

AngularJS进阶(十四)AngularJS路由问题分析

No Silver Bullet

路由 AngularJS 12月月更

easyrecovery2023手机版数据恢复软件下载

茶色酒

EasyRecovery EasyRecovery15 easyrecovery2023

C语言怎么给函数添加形参的默认值

eng八戒

c 函数 参数默认值

Linux 定时器介绍

eng八戒

c 定时器 Linux Kenel C++

看完这篇,还不懂JAVA内存模型(JMM)算我输

JAVA旭阳

Java 并发

AngularJS进阶(十六)脏值检查及语音识别集成

No Silver Bullet

语音识别 AngularJS 12月月更 脏值检查

架构实战-模块1作业

mm

架构实战营 10期

这才是Git的正确学习方式!

Jackpop

EasyRecovery2023个人免费版电脑数据恢复软件

茶色酒

EasyRecovery easyrecovery2023

cmake 入门笔记

eng八戒

c makefile 工程管理 cmake C++

太简单了,一文彻底搞懂Jenkins的用法!

Jackpop

日期格式化 YYYY-MM-DD 出现时间偏移量

HoneyMoose

刨析一下C++构造析构函数能不能声明为虚函数的背后机理?

eng八戒

继承 纯虚函数 C++

3月寒窗!啃透美团保姆级分布式进阶技术手册,4月终入美团定L8

程序知音

Java 分布式 java架构 后端技术

亚马逊 CTO Werner Vogels:2023 年及未来五大技术趋势预测

亚马逊云科技 (Amazon Web Services)

人工智能 大数据 亚马逊云科技

架构实战营模块1第2课 - 如何画出优秀的架构图

净意

Python初学者必备!适合新手阅读的Github开源代码。。。

Jackpop

Java架构师:ASM已经过时,建议使用类文件API简化开发_编程语言_A N M Bazlur Rahman_InfoQ精选文章