Javaassist 就是一个用来 处理 Java 字节码的类库。它可以在一个已经编译好的类中添加新的方法,或者是修改已有的方法,并且不需要对字节码方面有深入的了解。同时也可以去生成一个新的类对象,通过完全手动的方式。

首先需要引入jar包:

<dependency><groupId>org.javassist</groupId><artifactId>javassist</artifactId><version>3.25.0-GA</version></dependency>

示例:创建Person类的字节码


import javassist.*;public class CreatePerson {private static void createPerson() throws Exception {ClassPool pool = ClassPool.getDefault();// 1. 创建一个空类,类的全限定名CtClass cc =pool.makeClass("xxx.javassist.Person");// 2. 新增一个字段 private String name;// 字段名为nameCtField  fName = new CtField(pool.get("java.lang.String"), "name", cc);// 访问级别是 privatefName.setModifiers(Modifier.PRIVATE);//设置初始值 Tomcc.addField(fName, CtField.Initializer.constant("Tom"));//生成setter、getter方法cc.addMethod(CtNewMethod.setter("setName", fName));cc.addMethod(CtNewMethod.getter("getName", fName));// 4. 添加无参的构造函数CtConstructor cons = new CtConstructor(new CtClass[]{}, cc);cons.setBody("{name = \"Tom\";}");cc.addConstructor(cons);// 5. 添加有参的构造函数cons = new CtConstructor(new CtClass[]{pool.get("java.lang.String")}, cc);// $0=this / $1,$2,$3... 代表方法参数cons.setBody("$0.name = $1;");cc.addConstructor(cons);CtMethod  ctMethod = new CtMethod(CtClass.voidType, "printName", new CtClass[]{}, cc);ctMethod.setModifiers(Modifier.PUBLIC);ctMethod.setBody("{System.out.println(name);}");cc.addMethod(ctMethod);cc.writeFile("D:/***/src/main/java/");}public static void main(String[] args) {try {createPerson();} catch (Exception e) {e.printStackTrace();}}}

生成的.class类

//
// Source code recreated from a .class file by IntelliJ IDEA
// (powered by Fernflower decompiler)
//package xxx.javassist;public class Person {private String name = "Tom";public void setName(String var1) {this.name = var1;}public String getName() {return this.name;}public Person() {this.name = "Tom";}public Person(String var1) {this.name = var1;}public void printName() {System.out.println(this.name);}
}

在 Javassist 中,类 Javaassit.CtClass 表示 class 文件。一个 GtClass (编译时类)对象可以处理一个 class 文件,ClassPool是 CtClass 对象的容器。它按需读取类文件来构造 CtClass 对象,并且保存 CtClass 对象以便以后使用。

需要注意的是 ClassPool 会在内存中维护所有被它创建过的 CtClass,当 CtClass 数量过多时,会占用大量的内存,API中给出的解决方案是 有意识的调用CtClass的detach()方法以释放内存。

ClassPool需要关注的方法:

getDefault : 返回默认的ClassPool 是单例模式的,一般通过该方法创建我们的ClassPool;

appendClassPath, insertClassPath : 将一个ClassPath加到类搜索路径的末尾位置 或 插入到起始位置。通常通过该方法写入额外的类搜索路径,以解决多个类加载器环境中找不到类的尴尬;

toClass : 将修改后的CtClass加载至当前线程的上下文类加载器中,CtClass的toClass方法是通过调用本方法实现。需要注意的是一旦调用该方法,则无法继续修改已经被加载的class;

get , getCtClass : 根据类路径名获取该类的CtClass对象,用于后续的编辑。

CtClass需要关注的方法:

freeze : 冻结一个类,使其不可修改;

isFrozen : 判断一个类是否已被冻结;

prune : 删除类不必要的属性,以减少内存占用。调用该方法后,许多方法无法将无法正常使用,慎用;

defrost : 解冻一个类,使其可以被修改。如果事先知道一个类会被defrost, 则禁止调用 prune 方法;

detach : 将该class从ClassPool中删除;

writeFile : 根据CtClass生成 .class 文件;

toClass : 通过类加载器加载该CtClass。

上面我们创建一个新的方法使用了CtMethod类。CtMthod代表类中的某个方法,可以通过CtClass提供的API获取或者CtNewMethod新建,通过CtMethod对象可以实现对方法的修改。

CtMethod中的一些重要方法:

  1. insertBefore : 在方法的起始位置插入代码;
  2. insterAfter : 在方法的所有 return 语句前插入代码以确保语句能够被执行,除非遇到exception;
  3. insertAt : 在指定的位置插入代码;
  4. setBody : 将方法的内容设置为要写入的代码,当方法被 abstract修饰时,该修饰符被移除;
  5. make : 创建一个新的方法。

注意到在上面代码中的:setBody()的时候我们使用了一些符号:

Copy

// $0=this / $1,$2,$3... 代表方法参数

cons.setBody("{$0.name = $1;}");

具体还有很多的符号可以使用,但是不同符号在不同的场景下会有不同的含义,所以在这里就不在赘述,可以看javassist 的说明文档。http://www.javassist.org/tutorial/tutorial2.html

javassist用于生成动态代理

public class JavassistDemo {private String demoProperty = "demo-value"; // 字段// demoProperty字段对应的getter/setter方法public String getDemoProperty() {return demoProperty;}public void setDemoProperty(String demoProperty) {this.demoProperty = demoProperty;}// JavassistDemo的成员方法public void operation() {System.out.println("operation():" + this.demoProperty);}
}
public class JavassitMainDemo {public static void main(String[] args) throws Exception {// 创建ProxyFactory工厂实例,它负责动态生成JavassistDemo的子类ProxyFactory factory = new ProxyFactory();factory.setSuperclass(JavassistDemo.class);// 设置Filter,用于确定哪些方法调用需要被代理factory.setFilter(m -> {if (m.getName().equals("operation")) {return true;}return false;});// 设置拦截处理逻辑,被拦截的方法会执行MethodHandler中的逻辑factory.setHandler((self, thisMethod, proceed, args1) -> {System.out.println("before operation");Object result = proceed.invoke(self, args1);System.out.println("after operation");return result;});// 生成代理类,并根据代理类创建代理对象Class<?> c = factory.createClass();JavassistDemo javassistDemo = (JavassistDemo) c.newInstance();// 执行operation()方法时会被拦截,进而执行代理逻辑javassistDemo.operation();System.out.println(javassistDemo.getDemoProperty());}
}

打印:

before operation
operation():demo-value
after operation
demo-value

感谢:

https://www.cnblogs.com/rickiyang/p/11336268.html

javassist使用并生成动态代理相关推荐

  1. 深入理解JDK动态代理原理,使用javassist动手写一个动态代理框架

    文章目录 系列文章索引 一.动手实现一个动态代理框架 1.初识javassist 2.使用javassist实现一个动态代理框架 二.JDK动态代理 1.编码实现 2.基本原理 (1)getProxy ...

  2. abp vnext没有生成动态代理js代码

    解决方法: https://github.com/abpframework/abp/issues/1469 主要是因为mvc tired 结构abp里面生成的controller和mvc里面的cont ...

  3. Java动态代理机制详解(JDK 和CGLIB,Javassist,ASM)

    class文件简介及加载 Java编译器编译好Java文件之后,产生.class 文件在磁盘中.这种class文件是二进制文件,内容是只有JVM虚拟机能够识别的机器码.JVM虚拟机读取字节码文件,取出 ...

  4. JVM插桩之四:Java动态代理机制的对比(JDK和CGLIB,Javassist,ASM)

    一.class文件简介及加载 Java编译器编译好Java文件之后,产生.class 文件在磁盘中.这种class文件是二进制文件,内容是只有JVM虚拟机能够识别的机器码.JVM虚拟机读取字节码文件, ...

  5. Javassist实现JDK动态代理

    提到JDK动态代理,相信很多人并不陌生.然而,对于动态代理的实现原理,以及如何编码实现动态代理功能,可能知道的人就比较少了.接下一来,我们就一起来看看JDK动态代理的基本原理,以及如何通过Javass ...

  6. 使用反射生成 JDK 动态代理

    反射 反射赋予了我们在运行时分析类和执行类中方法的能力. 通过反射,可以获取任意一个类的所有属性和方法,还可以调用这些方法和属性. 使用反射生成 JDK 动态代理 JDK 动态代理只能为接口创建动态代 ...

  7. 动态代理竟然如此简单!

    点击上方蓝色"方志朋",选择"设为星标" 回复"666"获取独家整理的学习资料! 这篇文章我们来聊一下 Java 中的动态代理. 动态代理在 ...

  8. java中动态代理实现机制

    v前言: 代理模式是常用的java设计模式,它的特征是代理类与委托类有同样的接口,代理类主要负责为委托类预处理消息.过滤消息.把消息转发给委托类,以及事后处理消息等.代理类与委托类之间通常会存在关联关 ...

  9. spring的aop的动态代理机制都有哪些_Spring学习(4):Spring AOP

    Spring AOP说明 AOP(Aspect Oriented Pragraming)面向切面编程,AOP采用横向抽取机制,取代了传统纵向继承体系的重复性代码(性能监视.事务管理.安全检查.缓存). ...

最新文章

  1. VirtualBox安装Ubuntu详细过程
  2. java和python的web自动化有什么区别-Python和Java哪个更适合做自动化测试?
  3. itchat群机器人的自动画实现
  4. C运算符解析及优先级
  5. java bean 单例模式_单例模式 - Beans_bag - 博客园
  6. 真是,原来可以这样啊
  7. SQL手工注入入门级笔记(更新中)
  8. mysql 关联查询慢_mysql慢查询语句分析总结
  9. 【华为云技术分享】华为专家亲述:如何转型搞 AI?
  10. 【NOI OpenJudge】【1.1】编程基础之输入输出
  11. ( 转 ) CORS 有一次 OPTIONS 请求的原理
  12. 数据结构学习---有序链表的合并
  13. 社区发现算法之——Louvain
  14. dsp+c语言字符串,DSP中常用的C语言关键字
  15. 一篇吃透前置加加和后置加加(附练习题)
  16. Java 进制转换 代码
  17. 微信小程序性能优化实用建议
  18. 硬件接口之S/PDIF
  19. 开放式耳机哪个品牌好?南卡、韶音、索尼、Oladance开放式耳机推荐
  20. 隔离技术之MUX VLAN

热门文章

  1. 奥比中光3D视觉AI开放平台焕新上线,建设AI算力+算法+数据全链路
  2. css图片自适应裁剪
  3. 金蝶显示服务器,金蝶云桌面显示连接服务器
  4. Android TextView字体设置
  5. 软件测试模型——V模型
  6. TYD深度学习入门 第六章 递归神经网络
  7. ubuntu目录查看命令
  8. python魔法方法__new__(),__init__()
  9. HTTP请求错误400、401、402、403、404、405、406、407、412、414、500、501、502解析
  10. Bayesian Neural Network Recent Papers-贝叶斯神经网络相关研究文章