JAVA字节码操作——javassist

本文对比了BCEL、ASM、CGLIB和Javassist等字节码操作库的特点和适用场景,详细介绍了Javassist的基本使用和局限性,适合对字节码操作感兴趣的开发者阅读。

字节码优势

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

字节码操作库

字节码类库描述
BCELByte 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默认值为100
        ctClass.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教程

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值