Javassist讲解1(介绍,读写字节码)

Javassist讲解1(介绍,读写字节码)


介绍

javassist 使Java字节码操作变得简单,它是一个用于在Java中编辑字节码的类库;

它使Java程序能够在运行时定义一个新类,并在JVM加载类文件时对其进行修改

与其他类似的字节码编辑器不同,javassist提供了两个级别的API:

源代码级别和字节码级别

如果用户使用源级API,他们可以在不了解Java字节码规范的情况下编辑类文件。整个API仅使用Java语言的词汇设计,甚至可以以源代码的形式指定插入的字节码,Javassist会即时的进行编译。

另一方面,字节级API允许用户像其他编辑器一样直接编辑类文件


一、读写字节码

javassist是一个用于处理Java的字节码的类库。Java字节码存储在成为类文件的二进制文件中。每个类文件都包含一个Java类或接口

java 复制代码
	//获取Class池
	ClassPool pool = ClassPool.getDefault();
	//获取CtClass
	CtClass cc = pool.get("net.mooctest.Yest");
	//修改字节码
	cc.setSuperclass(pool.get("net.mooctest.YestFather"));
	//输出字节码
	cc.writeFile("D:\\桌面\\pair\\PairHeap2\\result1666497514964\\target\\classes");
  • 首先获取一个ClassPool对象,该对象控制使用Javassist修改字节码,ClassPool是一个表示类文件的对象容器
  • ClassPool 是本质一个CtClass 对象的map,使用类名作为键,调用get方法,搜索整个ClassPool,如果没有找到对应的对象,就新构造一个CtClass类对象,该对象被记录在map中,但是注意,这里说得是加载的对象,如果类名对应类不存在,会抛出异常
  • 获取到ClassPool对象后,就可以进行修改,这里的话讲其父类修改为了BaseObj,然后是cc.writeFile(),可以将修改后的字节码进行保存,也可以直接通过toBytecode()获取字节码

最后看到Class文件中可以发现本来没有继承的Yest类成功继承了YestFather类

1.如何创建新的类

  • ClassPool的makeClass方法可以帮助我们定义一个新的不存在的类
  • 但是makeClass()不能创建接口,创建接口需要调用makeInterface方法
java 复制代码
	ClassPool pool = ClassPool.getDefault();
	
	CtClass cc = pool.makeClass("net.mooctest.NewClass");
	
	cc.writeFile("D:\\桌面\\pair\\PairHeap2\\result1666497514964\\target\\classes");

可以看到成功创建了一个NewClass类

2.类冻结

  • 执行了writeFile 或者 toBytecode() 等输出字节码的方法之后,CtClass就处于了冻结的状态,在此状态下不允许修改。
java 复制代码
	ClassPool pool = ClassPool.getDefault();
	
	CtClass cc = pool.makeClass("net.mooctest.Yest");
	
	cc.writeFile("D:\\桌面\\pair\\PairHeap2\\result1666497514964\\target\\classes");
	
	cc.setSuperclass(pool.get("net.mooctest.YestFather"));

	如果在输出之后修改再次修改:
	>>>>>>>>
	Exception in thread "main" java.lang.RuntimeException:
	net.mooctest.Yest class is frozen
  • CtClass 调用 defrost 可以解冻,变为可修改的状态
  • 当 ClassPool.doPruning 属性设置为 true 的时候 ,冻结的类不能再次被解冻
java 复制代码
	ClassPool pool = ClassPool.getDefault();
	
	ClassPool.doPruning = true;

	调用解冻会提示:
	dofrost(): net.mooctest.Yest was pruned
  • doPruning 属性设置为 true ,某一个特定类需要解冻的时候,可以提前调用,这样就不会报错
java 复制代码
	CtClass cc = pool.makeClass("net.mooctest.Yest");
	cc.stopPruning(true);

相关推荐
我的golang之路果然有问题几秒前
积累的 java 找工作资源
java·笔记
毕设源码-朱学姐几秒前
【开题答辩全过程】以 基于Python的茶语店饮品管理系统的设计与实现为例,包含答辩的问题和答案
开发语言·python
Legendary_008几秒前
LDR6020:单C口可充可放电PD协议芯片,开启USB2.0数据传输新体验
c语言·开发语言
源代码•宸2 分钟前
Golang基础语法(go语言error、go语言defer、go语言异常捕获、依赖管理、Go Modules命令)
开发语言·数据库·后端·算法·golang·defer·recover
行者963 分钟前
Flutter适配OpenHarmony:高效数据筛选组件的设计与实现
开发语言·前端·flutter·harmonyos·鸿蒙
xwill*5 分钟前
wandb的使用方法,以navrl为例
开发语言·python·深度学习
编程大师哥10 分钟前
Java 常见异常(按「运行时 / 编译时」分类)
java·开发语言
SnrtIevg13 分钟前
Vavr 用户指南
java·后端
bybitq18 分钟前
Leetcode131题解 -Python-回溯+cache缓存
开发语言·python
怦怦蓝24 分钟前
IDEA 开发邮件发送功能:全流程报错解决方案汇总
java·ide·intellij-idea·发邮件