Java基础-字节码技术

上一篇 <<<代理模式(Proxy Pattern)
下一篇 >>>@Async失效之谜


Java字节码增强指的是在Java字节码生成之后,对其进行修改,增强其功能,可减少冗余代码,提高性能等。

应用场景

  • AOP技术
  • Lombok去除重复代码插件
  • 利用字节码操作类库动态修改class文件等

操作步骤

总原则在内存中获取到原来的字节码,然后通过一些工具(如 ASM,Javaasist)来修改它的byte[]数组,得到一个新的byte数组。
a、修改字节码
在JVM加载用户的Class时拦截并修改或者在运行时,使用Instrumentation.redefineClasses方法来替换掉原来的字节码
b、使字节码生效
自定义ClassLoader来加载修改后的字节码;

常用字节码操作类库

  • BCEL---Byte Code Engineering Library(BCEL),这是Apache Software Foundation的Jakarta项目的一部分。BCEL是Java classworking 广泛使用的一种框架,它可以让您深入jvm汇编语言进行类库操作的细节。BCEL与javassist有不同的处理字节码方法,BCEL在实际的jvm指令层次上进行操作(BCEL拥有丰富的jvm指令集支持) 而javassist所强调的是源代码级别的工作。
  • ASM----是一个轻量级Java字节码操作框架,直接涉及到JVM底层的操作和指令
    高性能,高质量CGLB生成类库的底层就是基于ASM实现的
  • javassist是一个开源的分析,编辑和创建Java字节码的类库。性能较ASM差,跟cglib差不多,但是使用简单。很多开源框架都在使用它。
    它的最外层的API和JAVA的反射包中的API颇为类似。
    它主要由CtClass,CtMethod,,以及CtField几个类组成。用以执行和JDK反射API中java.lang.Class,java.lang.reflect.Method,java.lang.reflect.Method.Field相同的操作。

Javassist优势

a、比反射开销小,性能高。(javassist性能高于反射,低于ASM)
b、可实现如下功能:
– 动态生成 新的类
– 动态改变某个类的结构(添加/删除/修改新的属性/方法)

Javassist局限性

a、不支持JDK5.0新语法(包括泛型、枚举)
b、不支持注解修改,但可以通过底层的javassist类来解决,具体参考:javassist.bytecode.annotation
c、不支持数组的初始化,如 String[]{"1","2"} ,除非只有数组的容量为 1
d、不支持内部类和匿名类
e、不支持 continue 和 break表达式。
f、对于继承关系,有些不支持。例如
class A {}
class B extends A {}
class C extends B {}

使用字节码创建文件

/**
 * 反编译结果:
 * package com.jgspx.entity;
 * public class User
 * {
 *     private String name;
 *     private Integer age;
 *
 *     public String getName() {
 *         return this.name;
 *     }
 *
 *     public User(final String s, final Integer n) {
 *         this.name = this.name;
 *         this.age = this.age;
 *     }
 * }
 *
 */
public class 使用字节码创建文件 {
    public static void main(String[] args) throws CannotCompileException, NotFoundException, IOException {
        ClassPool pool = ClassPool.getDefault();
        // 1.创建user类
        CtClass userClass = pool.makeClass("com.jarye.entity.User");
        // 2.创建name 和age属性
        CtField nameField = CtField.make(" private String name;", userClass);
        CtField ageField = CtField.make("  private Integer age;", userClass);
        // 3.添加属性
        userClass.addField(nameField);
        userClass.addField(ageField);
        // 4.创建方法
        CtMethod nameMethod = CtMethod.make("public String getName() {return name;}", userClass);
        // 5.添加方法
        userClass.addMethod(nameMethod);
        // 6.添加构造函数
        CtConstructor ctConstructor = new CtConstructor(
                new CtClass[] { pool.get("java.lang.String"), pool.get("java.lang.Integer") }, userClass);

        ctConstructor.setBody("    { this.name = name; this.age = age; }");
        userClass.addConstructor(ctConstructor);
        // 生成class文件
        userClass.writeFile("/Users/jarye/Downloads");
    }
}

动态修改字节码文件

/**
 * 代码中原方法:
 * public class User {
 *     private String name;
 *
 *     private Integer aget;
 * }
 *
 * 动态修改后的类信息反编译结果:
 * package com.jgspx.entity;
 *
 * public class User
 * {
 *     private String name;
 *     private Integer aget;
 *
 *     public void sum(final int n, final int n2) {
 *         System.out.println(new StringBuffer().append("sun:").append(n + n2).toString());
 *     }
 * }
 *
 * 执行结果:
 * 开启事务
 * sun:7
 * 提交事务
 *
 */
public class 动态修改字节码文件 {
    public static void main(String[] args) {
        try {
            ClassPool pool = ClassPool.getDefault();
            // 读取com.jgspx.entity.User
            CtClass userClass = pool.get("com.jarye.entity.User");
            CtMethod method = new CtMethod(CtClass.voidType, "sum", new CtClass[] { CtClass.intType, CtClass.intType },
                    userClass);
            method.setBody("{System.out.println(\"sun:\" + ($1 + $2));}");
            // 添加方法
            userClass.addMethod(method);
            userClass.writeFile("/Users/jarye/Downloads");
            // 动态执行方法
            Class clazz = userClass.toClass();
            Object newInstance = clazz.newInstance();

            Method sumMethod = clazz.getDeclaredMethod("sum", int.class, int.class);
            System.out.println("开启事务");
            sumMethod.invoke(newInstance, 2, 5);
            // 使用 javassist 实现动态代理。
            System.out.println("提交事务");
        } catch (Exception e) {
            e.printStackTrace();
        }
    }
}

相关文章链接:
<<<Java基础-反射机制
<<<Java基础-创建对象的方式汇总
<<<Java基础-对象布局
<<<Java基础-对象的引用类型
<<<Class文件分析一个类为啥最多支持65535个接口
<<<为什么重写equals还要重写hashcode方法
<<<如何自定义注解
<<<十大经典排序算法汇总-动画演示
<<<JDK8十大新特性

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 158,560评论 4 361
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 67,104评论 1 291
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 108,297评论 0 243
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 43,869评论 0 204
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 52,275评论 3 287
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 40,563评论 1 216
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 31,833评论 2 312
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 30,543评论 0 197
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 34,245评论 1 241
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 30,512评论 2 244
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 32,011评论 1 258
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 28,359评论 2 253
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 33,006评论 3 235
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 26,062评论 0 8
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 26,825评论 0 194
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 35,590评论 2 273
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 35,501评论 2 268

推荐阅读更多精彩内容