本文主要是介绍使用javassist动态为已有的类添加方法,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!
需求
现在有一个类Person,只有一个int age属性。
现在有一个需求,如下:
动态地为Person类新增increase方法,实现age++的效果。
javassist简介
javassist是一个用于处理java字节码的库,可以动态地修改已有的类的字节码。
javassist与jdk动态代理、cglib的比较
以下是我自己的理解,如有不对,请指正:
javassist动态地修改已有的类的字节码,不是通过生成代码,而是直接修改了类的字节码。这与jdk动态代理、cglib有本质的不同。
因为jdk动态代理是生成原始类的代理,而cglib是对方法进行拦截(gpt给出的方案是对方法进行拦截,我自己也做过实现,如果cglib场景中,Person类不去实现一个有increase方法的接口,使用反射调用的increase方法的时候,直接会报错)。
如果要通过jdk动态代理实现这个需求,则需要Person类实现一个接口,接口有increase方法,可以没有默认实现,否则jdk动态代理不能实现需求。
如果要通过cglib实现需求,则需要Person类实现一个接口,接口有默认方法increase,默认方法可以是空实现,否则cglib不能实现此需求。
javassist应用
前几天看rpc相关的文章,看到dubbo这个rpc是使用的javassist来生成接口的代理,对远程方法进行调用。
为啥dubbo使用javassist来生成带俩,而不是jdk动态代理呢?据说是因为javassis的性能更好,具体为啥更好,后面有空了,可以研究下。
代码
引入maven依赖
<dependencies><dependency><groupId>org.javassist</groupId><artifactId>javassist</artifactId><version>3.28.0-GA</version></dependency><dependency><groupId>org.projectlombok</groupId><artifactId>lombok</artifactId><version>1.18.30</version><scope>compile</scope></dependency>
</dependencies>
创建Person类
import lombok.Data;
import lombok.ToString;@Data
@ToString
public class Person {private int age;
}
测试代码
测试代码中,动态地为Person类新增increase方法,方法中将age+1。
import javassist.*;import java.lang.reflect.Method;public class DynamicMethodGeneratorTest {public static void main(String[] args) throws Exception {// 为Person动态添加方法dynamicAddMethodForPerson();Person person = new Person();// 设置age属性为20person.setAge(20);System.out.println("Before increase: " + person); // 20// 使用反射调用increase方法Method increaseMethod = person.getClass().getMethod("increase");increaseMethod.invoke(person);System.out.println("After increase: " + person); // 21}private static CtClass dynamicAddMethodForPerson() throws NotFoundException, CannotCompileException {ClassPool pool = ClassPool.getDefault();CtClass personClass = pool.get("com.qqcr.train.add_code.Person");// 生成一个新方法increase()CtMethod newMethod = CtNewMethod.make("public void increase() { age++; }", // 方法的作用是为age属性+1personClass);// 将increase方法添加到personClass中。personClass.addMethod(newMethod);/*这行代码必须要调用,调用之后,添加的increase方法才会生效。否则new Person().getClass().getMethod("increase")会报错:NoSuchMethodException。我猜是因为调用了toClass,才会用这里的personClass替换元空间中的Person.class。*/personClass.toClass();return personClass;}
}
测试代码最后的输出
Before increase: Person(age=20)
After increase: Person(age=21)
参考
CHATGTP-POE问答
这篇关于使用javassist动态为已有的类添加方法的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!