Java的动态性两种方式实现:字节码操作和反射
运行时操作字节码可以让我们实现以下功能:
- 动态生成新的类
- 动态改变某个类的结构(添加/删除/修改 新的属性/方法)
操作字节码的优势:
- 比反射开销小,性能高
- Javassist性能高于反射,低于ASM
而常见字节码操作类库:
此次,主要介绍Javassist相关内容:
Javassist
是一个开源的分析、编辑和创建Java字节码的类库,其主要的优点,在于简单,而且快速。直接使用java编码的形式,而不需要了解虚拟机指令,就能动态改变类的结构,或者动态生成类。
其中经常使用的方法:
ClassPool:javassist的类池,使用ClassPool 类可以跟踪和控制所操作的类,它的工作方式与 JVM 类装载器非常相似
CtClass: CtClass提供了类的操作,如在类中动态添加新字段、方法和构造函数、以及改变类、父类和接口的方法。
CtField:类的属性,通过它可以给类创建新的属性,还可以修改已有的属性的类型,访问修饰符等
CtMethod:类中的方法,通过它可以给类创建新的方法,还可以修改返回类型,访问修饰符等, 甚至还可以修改方法体内容代码
CtConstructor:与CtMethod类似
方法体内容代码 $1代表第一个参数,$2代表第二个参数
下面进行具体操作,使用Javassist创建类Emp:
import javassist.*;
public class Demo2 {
public static void main(String[] args) throws Exception {
ClassPool pool = ClassPool.getDefault();
CtClass cc = pool.makeClass("com.shsxt.bean.Emp");
//创建属性
CtField f1 = CtField.make("private int empno;",cc);
CtField f2 = CtField.make("private String ename;",cc);
cc.addField(f1);
cc.addField(f2);
//创建方法
CtMethod m1=CtMethod.make("public int getEmpno(){return empno;}",cc);
CtMethod m2=CtMethod.make("public void setEmpno(int empno){this.empno=empno;}",cc);
cc.addMethod(m1);
cc.addMethod(m2);
//添加构造器
CtConstructor constructor = new CtConstructor(new CtClass[]{
CtClass.intType,pool.get("java.lang.String"