javassist教程
转载 https://www.cnblogs.com/rickiyang/p/11336268.html
javassist使用全解析
Java 字节码以二进制的形式存储在 .class 文件中,每一个 .class 文件包含一个 Java 类或接口。Javaassist 就是一个用来 处理 Java 字节码的类库。它可以在一个已经编译好的类中添加新的方法,或者是修改已有的方法,并且不需要对字节码方面有深入的了解。同时也可以去生成一个新的类对象,通过完全手动的方式。
. 使用 Javassist 创建一个 class 文件#
首先需要引入jar包:
Copy
org.javassist
javassist
3.25.0-GA
编写创建对象的类:
Copy
package com.rickiyang.learn.javassist;
import javassist.*;
/**
@author rickiyang
@date 2019-08-06
@Desc
*/
public class CreatePerson {/**
创建一个Person 对象
@throws Exception
*/
public static void createPseson() throws Exception {
ClassPool pool = ClassPool.getDefault();// 1. 创建一个空类
CtClass cc = pool.makeClass(“com.rickiyang.learn.javassist.Person”);// 2. 新增一个字段 private String name;
// 字段名为name
CtField param = new CtField(pool.get(“java.lang.String”), “name”, cc);
// 访问级别是 private
param.setModifiers(Modifier.PRIVATE);
// 初始值是 “xiaoming”
cc.addField(param, CtField.Initializer.constant(“xiaoming”));// 3. 生成 getter、setter 方法
cc.addMethod(CtNewMethod.setter(“setName”, param));
cc.addMethod(CtNewMethod.getter(“getName”, param));// 4. 添加无参的构造函数
CtConstructor cons = new CtConstructor(new CtClass[]{}, cc);
cons.setBody("{name = “xiaohong”;}");
cc.addConstructor(cons);// 5. 添加有参的构造函数
cons = new CtConstructor(new CtClass[]{pool.get(“java.lang.String”)}, cc);
// $0=this / $1,$2,$3… 代表方法参数
cons.setBody("{$0.name = $1;}");
cc.addConstructor(cons);// 6. 创建一个名为printName方法,无参数,无返回值,输出name值
CtMethod ctMethod = new CtMethod(CtClass.voidType, “printName”, new CtClass[]{}, cc);
ctMethod.setModifiers(Modifier.PUBLIC);
ctMethod.setBody("{System.out.println(name);}");
cc.addMethod(ctMethod);//这里会将这个创建的类对象编译为.class文件
cc.writeFile("/Users/yangyue/workspace/springboot-learn/java-agent/src/main/java/");
}
public static void main(String[] args) {
try {
createPseson();
} catch (Exception e) {
e.printStackTrace();
}
}
}
执行上面的 main 函数之后,会在指定的目录内生成 Person.class 文件:
Copy
//
// Source code recreated from a .class file by IntelliJ IDEA
// (powered by Fernflower decompiler)
//
package com.rickiyang.learn.javassist;
public class Person {
private String name = “xiaoming”;
public void setName(String var1) {this.name = var1;
}public String getName() {return this.name;
}public Person() {this.name = "xiaohong";
}public Person(String var1) {this.name = var1;
}public void printName() {System.out.println(this.name);
}
}
跟咱们预想的一样。
在 Javassist 中,类 Javaassit.CtClass 表示 class 文件。一个 GtClass (编译时类)对象可以处理一个 class 文件,ClassPool是 CtClass 对象的容器。它按需读取类文件来构造 CtClass 对象,并且保存 CtClass 对象以便以后使用。
需要注意的是 ClassPool 会在内存中维护所有被它创建过的 CtClass,当 CtClass 数量过多时,会占用大量的内存,API中给出的解决方案是 有意识的调用CtClass的detach()方法以释放内存。
ClassPool需要关注的方法:
getDefault : 返回默认的ClassPool 是单例模式的,一般通过该方法创建我们的ClassPool;
appendClassPath, insertClassPath : 将一个ClassPath加到类搜索路径的末尾位置 或 插入到起始位置。通常通过该方法写入额外的类搜索路径,以解决多个类加载器环境中找不到类的尴尬;
toClass : 将修改后的CtClass加载至当前线程的上下文类加载器中,CtClass的toClass方法是通过调用本方法实现。需要注意的是一旦调用该方法,则无法继续修改已经被加载的class;
get , getCtClass : 根据类路径名获取该类的CtClass对象,用于后续的编辑。
CtClass需要关注的方法:
freeze : 冻结一个类,使其不可修改;
isFrozen : 判断一个类是否已被冻结;
prune : 删除类不必要的属性,以减少内存占用。调用该方法后,许多方法无法将无法正常使用,慎用;
defrost : 解冻一个类,使其可以被修改。如果事先知道一个类会被defrost, 则禁止调用 prune 方法;
detach : 将该class从ClassPool中删除;
writeFile : 根据CtClass生成 .class 文件;
toClass : 通过类加载器加载该CtClass。
上面我们创建一个新的方法使用了CtMethod类。CtMthod代表类中的某个方法,可以通过CtClass提供的API获取或者CtNewMethod新建,通过CtMethod对象可以实现对方法的修改。
CtMethod中的一些重要方法:
insertBefore : 在方法的起始位置插入代码;
insterAfter : 在方法的所有 return 语句前插入代码以确保语句能够被执行,除非遇到exception;
insertAt : 在指定的位置插入代码;
setBody : 将方法的内容设置为要写入的代码,当方法被 abstract修饰时,该修饰符被移除;
make : 创建一个新的方法。
注意到在上面代码中的:setBody()的时候我们使用了一些符号:
Copy
// $0=this / $1,$2,$3… 代表方法参数
cons.setBody("{$0.name = $1;}");
具体还有很多的符号可以使用,但是不同符号在不同的场景下会有不同的含义,所以在这里就不在赘述,可以看javassist 的说明文档。http://www.javassist.org/tutorial/tutorial2.html
- 调用生成的类对象#
- 通过反射的方式调用
上面的案例是创建一个类对象然后输出该对象编译完之后的 .class 文件。那如果我们想调用生成的类对象中的属性或者方法应该怎么去做呢?javassist也提供了相应的api,生成类对象的代码还是和第一段一样,将最后写入文件的代码替换为如下:
Copy
// 这里不写入文件,直接实例化
Object person = cc.toClass().newInstance();
// 设置值
Method setName = person.getClass().getMethod(“setName”, String.class);
setName.invoke(person, “cunhua”);
// 输出值
Method execute = person.getClass().getMethod(“printName”);
execute.invoke(person);
然后执行main方法就可以看到调用了 printName方法。
- 通过读取 .class 文件的方式调用
Copy
ClassPool pool = ClassPool.getDefault();
// 设置类路径
pool.appendClassPath("/Users/yangyue/workspace/springboot-learn/java-agent/src/main/java/");
CtClass ctClass = pool.get(“com.rickiyang.learn.javassist.Person”);
Object person = ctClass.toClass().newInstance();
// … 下面和通过反射的方式一样去使用 - 通过接口的方式
上面两种其实都是通过反射的方式去调用,问题在于我们的工程中其实并没有这个类对象,所以反射的方式比较麻烦,并且开销也很大。那么如果你的类对象可以抽象为一些方法得合集,就可以考虑为该类生成一个接口类。这样在newInstance()的时候我们就可以强转为接口,可以将反射的那一套省略掉了。
还拿上面的Person类来说,新建一个PersonI接口类:
Copy
package com.rickiyang.learn.javassist;
/**
@author rickiyang
@date 2019-08-07
@Desc
*/
public interface PersonI {void setName(String name);
String getName();
void printName();
}
实现部分的代码如下:
Copy
ClassPool pool = ClassPool.getDefault();
pool.appendClassPath("/Users/yangyue/workspace/springboot-learn/java-agent/src/main/java/");
// 获取接口
CtClass codeClassI = pool.get(“com.rickiyang.learn.javassist.PersonI”);
// 获取上面生成的类
CtClass ctClass = pool.get(“com.rickiyang.learn.javassist.Person”);
// 使代码生成的类,实现 PersonI 接口
ctClass.setInterfaces(new CtClass[]{codeClassI});
// 以下通过接口直接调用 强转
PersonI person = (PersonI)ctClass.toClass().newInstance();
System.out.println(person.getName());
person.setName(“xiaolv”);
person.printName();
使用起来很轻松。
- 修改现有的类对象#
前面说到新增一个类对象。这个使用场景目前还没有遇到过,一般会遇到的使用场景应该是修改已有的类。比如常见的日志切面,权限切面。我们利用javassist来实现这个功能。
有如下类对象:
Copy
package com.rickiyang.learn.javassist;
/**
@author rickiyang
@date 2019-08-07
@Desc
*/
public class PersonService {public void getPerson(){
System.out.println(“get Person”);
}public void personFly(){
System.out.println(“oh my god,I can fly”);
}
}
然后对他进行修改:
Copy
package com.rickiyang.learn.javassist;
import javassist.ClassPool;
import javassist.CtClass;
import javassist.CtMethod;
import javassist.Modifier;
import java.lang.reflect.Method;
/**
@author rickiyang
@date 2019-08-07
@Desc
*/
public class UpdatePerson {public static void update() throws Exception {
ClassPool pool = ClassPool.getDefault();
CtClass cc = pool.get(“com.rickiyang.learn.javassist.PersonService”);CtMethod personFly = cc.getDeclaredMethod("personFly");personFly.insertBefore("System.out.println(\"起飞之前准备降落伞\");");personFly.insertAfter("System.out.println(\"成功落地。。。。\");");//新增一个方法CtMethod ctMethod = new CtMethod(CtClass.voidType, "joinFriend", new CtClass[]{}, cc);ctMethod.setModifiers(Modifier.PUBLIC);ctMethod.setBody("{System.out.println(\"i want to be your friend\");}");cc.addMethod(ctMethod);Object person = cc.toClass().newInstance();// 调用 personFly 方法Method personFlyMethod = person.getClass().getMethod("personFly");personFlyMethod.invoke(person);//调用 joinFriend 方法Method execute = person.getClass().getMethod("joinFriend");execute.invoke(person);
}
public static void main(String[] args) {
try {
update();
} catch (Exception e) {
e.printStackTrace();
}
}
}
在personFly方法前后加上了打印日志。然后新增了一个方法joinFriend。执行main函数可以发现已经添加上了。
另外需要注意的是:上面的insertBefore() 和 setBody()中的语句,如果你是单行语句可以直接用双引号,但是有多行语句的情况下,你需要将多行语句用{}括起来。javassist只接受单个语句或用大括号括起来的语句块。
javassist教程相关推荐
- Spring Hibernate教程
1.简介 在本文中,我们将演示如何利用最流行的ORM(对象关系映射)工具之一的Hibernate的功能 ,该工具可将面向对象的域模型转换为传统的关系数据库. Hibernate是目前最流行的Java框 ...
- Spring休眠教程
1.简介 在本文中,我们将演示如何利用最流行的ORM(对象关系映射)工具之一的Hibernate的功能 ,该工具可将面向对象的域模型转换为传统的关系数据库. Hibernate是目前最流行的Java框 ...
- HQL - Hibernate查询语言 - 示例教程
HQL - Hibernate查询语言 - 示例教程 HQL或Hibernate查询语言是Hibernate Framework的面向对象查询语言.HQL与SQL非常相似,只是我们使用Objects而 ...
- Hibernate Criteria示例教程
Hibernate Criteria示例教程 欢迎使用Hibernate Criteria示例教程.今天我们将研究Hibernate中的Criteria. Hibernate Criteria 大多数 ...
- Hibernate初学者教程
Hibernate初学者教程 欢迎来到Hibernate初学者教程.Hibernate是最广泛使用的Java ORM工具之一.大多数应用程序使用关系数据库来存储应用程序信息,在较低级别,我们使用JDB ...
- Primefaces Spring和Hibernate集成示例教程
Primefaces Spring和Hibernate集成示例教程 欢迎使用Spring Primefaces和Hibernate Integration示例.框架之间的集成是一项复杂的任务,而且大多 ...
- 菜鸟学习笔记:Java提升篇12(Java动态性2——动态编译、javassist字节码操作)
菜鸟学习笔记:Java提升篇12(Java动态性2--动态编译.javassist字节码操作) Java的动态编译 通过脚本引擎执行代码 Java字节码操作 JAVAssist的简单使用 常用API ...
- matlab初学者教程_初学者的Hibernate教程
matlab初学者教程 Welcome to the Hibernate tutorial for Beginners. Hibernate is one of the most widely use ...
- hibernate示例_Hibernate条件示例教程
hibernate示例 Welcome to the Hibernate Criteria Example Tutorial. Today we will look into Criteria in ...
- primefaces教程_Primefaces Spring和Hibernate集成示例教程
primefaces教程 Welcome to the Spring Primefaces and Hibernate Integration example. Integration between ...
最新文章
- python中将HTTP头部中的GMT时间转换成datetime时间格式
- express项目创建
- CVPR2021 | 深度解读RepVGG!
- 一个漂亮而强大的RecyclerView
- 基于Python+Django的在线习题考试测试管理系统
- The Turn Model for Adaptive Routing中的west-first算法
- 一文看懂搜狗招股书:90次提到AI,王小川持股5%,净利3.7亿
- C#实现拖放获取文件路径
- 三、java语法基础
- 输入上标的html标签是,HTML上标sup与下标注sub标签元素
- 智能图形开发板SmartGLCD
- Spring事务原理分析(三)--事务代理调用过程
- 货物与产品的区别_商品与产物的区别
- 福禄克FLUKE OFP2-100-Q CH与OFP2-Q-ADD技术规格
- 恢复Redis被误删的数据
- 常用数据库URL地址的写法
- linux内核启动流程(文章最后流程图)
- beeline 查询表数据导出到本地文件csv
- Ubuntu16.04安装搜狗拼音输入法(中文输入法)[转]
- S32K14x CAN休眠唤醒的实现方案
热门文章
- Cocos2d-x建工程时避免copy文件夹和库(子龙山人)
- 配置管理工具SVN的使用
- VS连接SQL Server 2008,并实现登录和注册功能
- 月薪30K+的电子工程师应具备什么?
- 《东周列国志》第九十回 苏秦合纵相六国 张仪被激往秦邦
- Alexa | Alexa Auto SDK 概述
- android studio布局文件在哪,android studio布局文件详解
- php进度条实例,JavaScript_一个简单的jquery进度条示例,用jQuery实现的最简单的进度条 - phpStudy...
- matlab傅里叶变换程序
- 关于foobar2000中Convolver,大家觉得哪个Impulse效果最好?