本文主要是介绍JAVA字节码操作——javassist,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!
文章目录
- 字节码优势
- 字节码操作库
- Javassist主要类
- Javassist占位符
- Javasist入门使用
- Javassist局限性
- 详细教程参考
字节码优势
- 比反射开销小,性能高
字节码操作库
字节码类库 | 描述 |
---|---|
BCEL | Byte Code Engineering Library (BCEL) ,这是 Apache Software Foundation 的 Jakarta 项目的一分。 BCEL 是 Java classworking 广泛 使用的一种 框架 , 它 可以让您深入 JVM 汇编语言进行类操作的细节。 BCEL 与Javassist 有不同的处理字节码方法, BCEL 在实际的 JVM 指令层次上进行操作 (BCEL 拥有丰富的 JVM 指令级支持 ) 而Javassist 所 强调 的是源代码 级别的工作(性能高,学习成本高) |
ASM | 是一个轻量级 java 字节码操作框架,直接涉及到 JVM 底层的操作和 指令(性能高,学习成本高) |
CGLIB(Code Generation Library) | 是一个强大的,高性能,高质量的 Code 生成类库,基于 ASM 实现。(性能一般,学习成本低) |
Javassist | 是一个开源的分析、编辑和创建 Java 字节码 的类库 。性能较 ASM 差,跟 cglib 差不多,但是使用简单。很多开源框架都在使用它(性能一般,学习成本低) |
Javassist主要类
CtClass,CtMethod,CtField,CtConstruct几个类组成,对应JDK中的反射为java.lang.Class,java.lang.reflect.Method和java.lang.reflect.Method.Field
Javassist占位符
Javasist入门使用
public static void main(String[] args) throws Exception {// 创建类池ClassPool pool = ClassPool.getDefault();// 创建类CtClass ctClass = pool.makeClass("com.sean.dto.Person");// 创建属性CtField cfName = CtField.make("private String name;", ctClass);CtField cfAge = CtField.make("private int age;", ctClass);ctClass.addField(cfName);// 设置age默认值为100ctClass.addField(cfAge, "100");// 创建方法CtMethod setName = CtMethod.make("public void setName(String name){this.name=name;}", ctClass);CtMethod getName = CtMethod.make("public String getName(){return this.name;}", ctClass);CtMethod setAge = CtMethod.make("public void setAge(int age){this.age=age;}", ctClass);CtMethod getAge = CtMethod.make("public int getAge(){return this.age;}", ctClass);CtMethod staticGetMax = CtMethod.make("public static int getMax(int a,int b){return a>b?a:b;}", ctClass);// 在获取最大值方法前加入方法staticGetMax.insertBefore("System.out.println($1+\" and \"+$2+\" start\");");ctClass.addMethod(setName);ctClass.addMethod(getName);ctClass.addMethod(setAge);ctClass.addMethod(getAge);ctClass.addMethod(staticGetMax);// 创建构造器CtConstructor ctConstructor = new CtConstructor(new CtClass[]{pool.get("java.lang.String"), CtClass.intType}, ctClass);ctConstructor.setBody("{this.name=$1;this.age=$2;}");ctConstructor.insertBefore("System.out.println(\"this is constructor\");");ctClass.addConstructor(ctConstructor);// 输出字节码文件ctClass.writeFile("E:\\桌面\\javassist_code");System.out.println("代码生成成功");// 实例化创建的字节类Class clazz = ctClass.toClass();Constructor constructor = clazz.getConstructor(String.class, int.class);Object obj = constructor.newInstance("zhangsan", 11);// 进行方法调用Method getMax = clazz.getDeclaredMethod("getMax", int.class, int.class);Object invoke = getMax.invoke(obj, 12, 88);System.out.println("get 12 and 88 max: "+invoke);// 进行属性获取Field age = clazz.getDeclaredField("age");age.setAccessible(true);Object o = age.get(obj);System.out.println("age: " + o);}
Javassist局限性
- 不支持泛型、枚举
- 不支持数组的初始化,如String[]{“1”,“2”},除非只有数组的容量为1
- 不支持内部类和匿名类
- 不支持continue和break表达式
- 对于继承关系,有些不支持。如:
class A{}
class B extends A{}
class C extends B{}
详细教程参考
javassist教程
这篇关于JAVA字节码操作——javassist的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!