Java 字节码以二进制的形式存储在 .class 文件中,每一个 .class 文件包含一个 Java 类或接口。Javaassist 就是一个用来 处理 Java 字节码的类库。它可以在一个已经编译好的类中添加新的方法,或者是修改已有的方法,并且不需要对字节码方面有深入的了解。同时也可以去生成一个新的类对象,通过完全手动的方式。

1. 使用 Javassist 创建一个 class 文件

首先需要引入jar包:

  1. <dependency>
  2. <groupId>org.javassist</groupId>
  3. <artifactId>javassist</artifactId>
  4. <version>3.25.0-GA</version>
  5. </dependency>

编写创建对象的类:

  1. package com.rickiyang.learn.javassist;
  2. import javassist.*;
  3. /**
  4. * @author rickiyang
  5. * @date 2019-08-06
  6. * @Desc
  7. */
  8. public class CreatePerson {
  9. /**
  10. * 创建一个Person 对象
  11. *
  12. * @throws Exception
  13. */
  14. public static void createPseson() throws Exception {
  15. ClassPool pool = ClassPool.getDefault();
  16. // 1. 创建一个空类
  17. CtClass cc = pool.makeClass("com.rickiyang.learn.javassist.Person");
  18. // 2. 新增一个字段 private String name;
  19. // 字段名为name
  20. CtField param = new CtField(pool.get("java.lang.String"), "name", cc);
  21. // 访问级别是 private
  22. param.setModifiers(Modifier.PRIVATE);
  23. // 初始值是 "xiaoming"
  24. cc.addField(param, CtField.Initializer.constant("xiaoming"));
  25. // 3. 生成 getter、setter 方法
  26. cc.addMethod(CtNewMethod.setter("setName", param));
  27. cc.addMethod(CtNewMethod.getter("getName", param));
  28. // 4. 添加无参的构造函数
  29. CtConstructor cons = new CtConstructor(new CtClass[]{}, cc);
  30. cons.setBody("{name = \"xiaohong\";}");
  31. cc.addConstructor(cons);
  32. // 5. 添加有参的构造函数
  33. cons = new CtConstructor(new CtClass[]{pool.get("java.lang.String")}, cc);
  34. // $0=this / $1,$2,$3... 代表方法参数
  35. cons.setBody("{$0.name = $1;}");
  36. cc.addConstructor(cons);
  37. // 6. 创建一个名为printName方法,无参数,无返回值,输出name值
  38. CtMethod ctMethod = new CtMethod(CtClass.voidType, "printName", new CtClass[]{}, cc);
  39. ctMethod.setModifiers(Modifier.PUBLIC);
  40. ctMethod.setBody("{System.out.println(name);}");
  41. cc.addMethod(ctMethod);
  42. //这里会将这个创建的类对象编译为.class文件
  43. cc.writeFile("/Users/yangyue/workspace/springboot-learn/java-agent/src/main/java/");
  44. }
  45. public static void main(String[] args) {
  46. try {
  47. createPseson();
  48. } catch (Exception e) {
  49. e.printStackTrace();
  50. }
  51. }
  52. }

执行上面的 main 函数之后,会在指定的目录内生成 Person.class 文件:

  1. //
  2. // Source code recreated from a .class file by IntelliJ IDEA
  3. // (powered by Fernflower decompiler)
  4. //
  5. package com.rickiyang.learn.javassist;
  6. public class Person {
  7. private String name = "xiaoming";
  8. public void setName(String var1) {
  9. this.name = var1;
  10. }
  11. public String getName() {
  12. return this.name;
  13. }
  14. public Person() {
  15. this.name = "xiaohong";
  16. }
  17. public Person(String var1) {
  18. this.name = var1;
  19. }
  20. public void printName() {
  21. System.out.println(this.name);
  22. }
  23. }

跟咱们预想的一样。

在 Javassist 中,类 Javaassit.CtClass 表示 class 文件。一个 GtClass (编译时类)对象可以处理一个 class 文件,ClassPoolCtClass 对象的容器。它按需读取类文件来构造 CtClass 对象,并且保存 CtClass 对象以便以后使用。

需要注意的是 ClassPool 会在内存中维护所有被它创建过的 CtClass,当 CtClass 数量过多时,会占用大量的内存,API中给出的解决方案是 有意识的调用CtClassdetach()方法以释放内存

ClassPool需要关注的方法:

  1. getDefault : 返回默认的ClassPool 是单例模式的,一般通过该方法创建我们的ClassPool;
  2. appendClassPath, insertClassPath : 将一个ClassPath加到类搜索路径的末尾位置 或 插入到起始位置。通常通过该方法写入额外的类搜索路径,以解决多个类加载器环境中找不到类的尴尬;
  3. toClass : 将修改后的CtClass加载至当前线程的上下文类加载器中,CtClass的toClass方法是通过调用本方法实现。需要注意的是一旦调用该方法,则无法继续修改已经被加载的class
  4. get , getCtClass : 根据类路径名获取该类的CtClass对象,用于后续的编辑。

CtClass需要关注的方法:

  1. freeze : 冻结一个类,使其不可修改;
  2. isFrozen : 判断一个类是否已被冻结;
  3. prune : 删除类不必要的属性,以减少内存占用。调用该方法后,许多方法无法将无法正常使用,慎用;
  4. defrost : 解冻一个类,使其可以被修改。如果事先知道一个类会被defrost, 则禁止调用 prune 方法;
  5. detach : 将该class从ClassPool中删除;
  6. writeFile : 根据CtClass生成 .class 文件;
  7. toClass : 通过类加载器加载该CtClass。

上面我们创建一个新的方法使用了CtMethod类。CtMthod代表类中的某个方法,可以通过CtClass提供的API获取或者CtNewMethod新建,通过CtMethod对象可以实现对方法的修改。

CtMethod中的一些重要方法:

  1. insertBefore : 在方法的起始位置插入代码;
  2. insterAfter : 在方法的所有 return 语句前插入代码以确保语句能够被执行,除非遇到exception;
  3. insertAt : 在指定的位置插入代码;
  4. setBody : 将方法的内容设置为要写入的代码,当方法被 abstract修饰时,该修饰符被移除;
  5. make : 创建一个新的方法。

注意到在上面代码中的:setBody()的时候我们使用了一些符号:

  1. // $0=this / $1,$2,$3... 代表方法参数
  2. cons.setBody("{$0.name = $1;}");

具体还有很多的符号可以使用,但是不同符号在不同的场景下会有不同的含义,所以在这里就不在赘述,可以看javassist 的说明文档。http://www.javassist.org/tutorial/tutorial2.html

2. 调用生成的类对象

1. 通过反射的方式调用

上面的案例是创建一个类对象然后输出该对象编译完之后的 .class 文件。那如果我们想调用生成的类对象中的属性或者方法应该怎么去做呢?javassist也提供了相应的api,生成类对象的代码还是和第一段一样,将最后写入文件的代码替换为如下:

  1. // 这里不写入文件,直接实例化
  2. Object person = cc.toClass().newInstance();
  3. // 设置值
  4. Method setName = person.getClass().getMethod("setName", String.class);
  5. setName.invoke(person, "cunhua");
  6. // 输出值
  7. Method execute = person.getClass().getMethod("printName");
  8. execute.invoke(person);

然后执行main方法就可以看到调用了 printName方法。

2. 通过读取 .class 文件的方式调用
  1. ClassPool pool = ClassPool.getDefault();
  2. // 设置类路径
  3. pool.appendClassPath("/Users/yangyue/workspace/springboot-learn/java-agent/src/main/java/");
  4. CtClass ctClass = pool.get("com.rickiyang.learn.javassist.Person");
  5. Object person = ctClass.toClass().newInstance();
  6. // ...... 下面和通过反射的方式一样去使用
3. 通过接口的方式

上面两种其实都是通过反射的方式去调用,问题在于我们的工程中其实并没有这个类对象,所以反射的方式比较麻烦,并且开销也很大。那么如果你的类对象可以抽象为一些方法得合集,就可以考虑为该类生成一个接口类。这样在newInstance()的时候我们就可以强转为接口,可以将反射的那一套省略掉了。

还拿上面的Person类来说,新建一个PersonI接口类:

  1. package com.rickiyang.learn.javassist;
  2. /**
  3. * @author rickiyang
  4. * @date 2019-08-07
  5. * @Desc
  6. */
  7. public interface PersonI {
  8. void setName(String name);
  9. String getName();
  10. void printName();
  11. }

实现部分的代码如下:

  1. ClassPool pool = ClassPool.getDefault();
  2. pool.appendClassPath("/Users/yangyue/workspace/springboot-learn/java-agent/src/main/java/");
  3. // 获取接口
  4. CtClass codeClassI = pool.get("com.rickiyang.learn.javassist.PersonI");
  5. // 获取上面生成的类
  6. CtClass ctClass = pool.get("com.rickiyang.learn.javassist.Person");
  7. // 使代码生成的类,实现 PersonI 接口
  8. ctClass.setInterfaces(new CtClass[]{codeClassI});
  9. // 以下通过接口直接调用 强转
  10. PersonI person = (PersonI)ctClass.toClass().newInstance();
  11. System.out.println(person.getName());
  12. person.setName("xiaolv");
  13. person.printName();

使用起来很轻松。

2. 修改现有的类对象

前面说到新增一个类对象。这个使用场景目前还没有遇到过,一般会遇到的使用场景应该是修改已有的类。比如常见的日志切面,权限切面。我们利用javassist来实现这个功能。

有如下类对象:

  1. package com.rickiyang.learn.javassist;
  2. /**
  3. * @author rickiyang
  4. * @date 2019-08-07
  5. * @Desc
  6. */
  7. public class PersonService {
  8. public void getPerson(){
  9. System.out.println("get Person");
  10. }
  11. public void personFly(){
  12. System.out.println("oh my god,I can fly");
  13. }
  14. }

然后对他进行修改:

  1. package com.rickiyang.learn.javassist;
  2. import javassist.ClassPool;
  3. import javassist.CtClass;
  4. import javassist.CtMethod;
  5. import javassist.Modifier;
  6. import java.lang.reflect.Method;
  7. /**
  8. * @author rickiyang
  9. * @date 2019-08-07
  10. * @Desc
  11. */
  12. public class UpdatePerson {
  13. public static void update() throws Exception {
  14. ClassPool pool = ClassPool.getDefault();
  15. CtClass cc = pool.get("com.rickiyang.learn.javassist.PersonService");
  16. CtMethod personFly = cc.getDeclaredMethod("personFly");
  17. personFly.insertBefore("System.out.println(\"起飞之前准备降落伞\");");
  18. personFly.insertAfter("System.out.println(\"成功落地。。。。\");");
  19. //新增一个方法
  20. CtMethod ctMethod = new CtMethod(CtClass.voidType, "joinFriend", new CtClass[]{}, cc);
  21. ctMethod.setModifiers(Modifier.PUBLIC);
  22. ctMethod.setBody("{System.out.println(\"i want to be your friend\");}");
  23. cc.addMethod(ctMethod);
  24. Object person = cc.toClass().newInstance();
  25. // 调用 personFly 方法
  26. Method personFlyMethod = person.getClass().getMethod("personFly");
  27. personFlyMethod.invoke(person);
  28. //调用 joinFriend 方法
  29. Method execute = person.getClass().getMethod("joinFriend");
  30. execute.invoke(person);
  31. }
  32. public static void main(String[] args) {
  33. try {
  34. update();
  35. } catch (Exception e) {
  36. e.printStackTrace();
  37. }
  38. }
  39. }

personFly方法前后加上了打印日志。然后新增了一个方法joinFriend。执行main函数可以发现已经添加上了。

另外需要注意的是:上面的insertBefore()setBody()中的语句,如果你是单行语句可以直接用双引号,但是有多行语句的情况下,你需要将多行语句用{}括起来。javassist只接受单个语句或用大括号括起来的语句块。

javassist使用全解析的更多相关文章

  1. Google Maps地图投影全解析(3):WKT形式表示

    update20090601:EPSG对该投影的编号设定为EPSG:3857,对应的WKT也发生了变化,下文不再修改,相对来说格式都是那样,可以到http://www.epsg-registry.or ...

  2. C#系统缓存全解析(转载)

    C#系统缓存全解析 对各种缓存的应用场景和方法做了很详尽的解读,这里推荐一下 转载地址:http://blog.csdn.net/wyxhd2008/article/details/8076105

  3. 【凯子哥带你学Framework】Activity界面显示全解析

    前几天凯子哥写的Framework层的解析文章<Activity启动过程全解析>,反响还不错,这说明“写让大家都能看懂的Framework解析文章”的思想是基本正确的. 我个人觉得,深入分 ...

  4. iOS Storyboard全解析

    来源:http://iaiai.iteye.com/blog/1493956 Storyboard)是一个能够节省你很多设计手机App界面时间的新特性,下面,为了简明的说明Storyboard的效果, ...

  5. 【转载】Fragment 全解析(1):那些年踩过的坑

    http://www.jianshu.com/p/d9143a92ad94 Fragment系列文章:1.Fragment全解析系列(一):那些年踩过的坑2.Fragment全解析系列(二):正确的使 ...

  6. (转)ASP.NET缓存全解析6:数据库缓存依赖

    ASP.NET缓存全解析文章索引 ASP.NET缓存全解析1:缓存的概述 ASP.NET缓存全解析2:页面输出缓存 ASP.NET缓存全解析3:页面局部缓存 ASP.NET缓存全解析4:应用程序数据缓 ...

  7. jQuery&nbsp;Ajax&nbsp;实例&nbsp;全解析

    jQuery Ajax 实例 全解析 jQuery确实是一个挺好的轻量级的JS框架,能帮助我们快速的开发JS应用,并在一定程度上改变了我们写JavaScript代码的习惯. 废话少说,直接进入正题,我 ...

  8. ARM内核全解析,从ARM7,ARM9到Cortex-A7,A8,A9,A12,A15到Cortex-A53,A57

    转自: ARM内核全解析,从ARM7,ARM9到Cortex-A7,A8,A9,A12,A15到Cortex-A53,A57 前不久ARM正式宣布推出新款ARMv8架构的Cortex-A50处理器系列 ...

  9. jQuery Ajax 全解析

    转自:http://www.cnblogs.com/qleelulu/archive/2008/04/21/1163021.html 本文地址: jQuery Ajax 全解析 本文作者:QLeelu ...

随机推荐

  1. testNG 注释实例

    1. 单个测试用例文件 新建TestDBConnection.java文件 import org.testng.annotations.*; public class TestDBConnection ...

  2. iOS学习——iOS项目增加新的字体

    基本思路 在项目开发过程中,iOS系统自带的字体库可能不适应需求,需要导入其他的字体库.下面是iOS项目增加新的字体的基本思路,基本上分为三步: 将字体库添加到项目中 在info.plist中添加所需 ...

  3. Hybris产品主数据的价格折扣维护

    登录Hybris backoffice的产品管理界面,进入price标签页,点击Create new Discount Row按钮: 在Discount下拉地段里选择10%的折扣,这个产品原来的单价是 ...

  4. 【RAC】将单实例备份集恢复为rac数据库

    [RAC]将单实例备份集恢复为rac数据库 一.1  BLOG文档结构图 一.2  前言部分 一.2.1  导读 各位技术爱好者,看完本文后,你可以掌握如下的技能,也可以学到一些其它你所不知道的知识, ...

  5. The Sum of the k-th Powers(Educational Codeforces Round 7F+拉格朗日插值法)

    题目链接 传送门 题面 题意 给你\(n,k\),要你求\(\sum\limits_{i=1}^{n}i^k\)的值. 思路 根据数学知识或者说题目提示可知\(\sum\limits_{i=1}^{n ...

  6. janusgraph-遍历图的语言

    精确查询 语句含义 测试语句 执行时间 查询顶点标签为FALV的顶点数量 g.V().hasLabel('FALV').count() 2400s 查询顶点属性中id为19012201 clockWi ...

  7. Django REST framework版本控制

    参考链接:https://www.cnblogs.com/liwenzhou/p/10269268.html 1.路由: #版本控制 re_path('^(?P<version>[v1|v ...

  8. LeetCode 873. Length of Longest Fibonacci Subsequence

    原题链接在这里:https://leetcode.com/problems/length-of-longest-fibonacci-subsequence/ 题目: A sequence X_1, X ...

  9. LeetCode 826. Most Profit Assigning Work

    原题链接在这里:https://leetcode.com/problems/most-profit-assigning-work/ 题目: We have jobs: difficulty[i] is ...

  10. Configure JSON.NET to ignore DataContract/DataMember attributes

    https://stackoverflow.com/questions/11055225/configure-json-net-to-ignore-datacontract-datamember-at ...