当前位置: 代码迷 >> 综合 >> JAVA字节码操作——javassist
  详细解决方案

JAVA字节码操作——javassist

热度:51   发布时间:2024-01-16 03:31:01.0

文章目录

  • 字节码优势
  • 字节码操作库
  • Javassist主要类
  • Javassist占位符
  • Javasist入门使用
  • Javassist局限性
  • 详细教程参考

字节码优势

  1. 比反射开销小,性能高

字节码操作库

字节码类库 描述
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教程

  相关解决方案