javassist简介
javassist可以对一个已经编译好了的.class文件的字节码进行改动,比如说我可以为一个类添加一个方法,添加一个属性,也可以修改一个方法等,还可以对一个方法,异常进行拦截等。
我们常用到的动态特性主要是反射,在运行时查找对象属性、方法,修改作用域,通过方法名称调用方法等。但反射的性能开销较大,Javassit是一个东京研究院开发的第三方库,提供了运行时操作Java字节码的方法。类似字节码操作方法还有ASM,几种动态编程方法相比较,在性能上Javassist高于反射,但低于ASM。
学习javassit,主要是为学习如何让已经运行中的虚拟机动态加载类做铺垫。
例子
maven 依赖
<dependency>
<groupId>org.javassist</groupId>
<artifactId>javassist</artifactId>
<version>3.20.0-GA</version>
</dependency>
java类
Screen类
package demo7;
/**
* Created by enyilr on 15/10/14.
*/
public class Screen {
public void plus(int i, int j) {
int m = i + j;
System.out.println("i plus j=" + m);
}
}
TransClass类
package demo5;
/**
* Created by enyilr on 15/10/14.
*/
public class TransClass {
public int getNumber() {
return 2;
}
}
测试类Test
public class Test {
public static void main(String[] args) throws NotFoundException, CannotCompileException, IOException, SecurityException, NoSuchMethodException, IllegalArgumentException, IllegalAccessException, InvocationTargetException, InstantiationException {
ClassPool pool = ClassPool.getDefault();
CtClass transClass = pool.get("demo5.TransClass");
CtMethod ctm = transClass.getDeclaredMethod("getNumber");
ctm.insertAfter("System.out.println(\"add down!\");");
transClass.writeFile();
Method m = transClass.toClass().getMethod("getNumber");
TransClass transClass1 = TransClass.class.newInstance();
Object invoke = m.invoke(transClass1);
System.out.println(invoke);
pool.importPackage("java.util.Date");
CtClass screen = pool.get("demo7.Screen");
CtMethod plus = screen.getDeclaredMethod("plus");
plus.insertBefore("{System.out.println(\"i=\"+($1)+\",j=\"+$2);}");
plus.insertAfter("System.out.println((new Date()));");
screen.writeFile();
Method plus1 = screen.toClass().getMethod("plus", int.class,int.class);
Screen screen1 = Screen.class.newInstance();
plus1.invoke(screen1,1,2);
}
}