詳解Java中Javassist的使用
開(kāi)篇
說(shuō)起 AOP 小伙伴們肯定很熟悉,無(wú)論是 JDK 動(dòng)態(tài)代理或者是 CGLIB 等,其底層都是通過(guò)操作 Java 字節(jié)碼來(lái)實(shí)現(xiàn)代理。常用的一些操作字節(jié)碼的技術(shù)有 ASM、AspectJ、Javassist 等。
ASM 其設(shè)計(jì)和實(shí)現(xiàn)是盡可能小而且快,更專注于性能。它在指令的層面來(lái)操作,所以使用它需要對(duì) JVM 的指令有所了解,門檻較高,CGLIB 就使用了 ASM 技術(shù)。
AspectJ 擴(kuò)展了 Java 語(yǔ)言,定義了一系列 AOP 語(yǔ)法,在 JVM 中運(yùn)行需要使用特定的編譯器生成遵守 Java 字節(jié)碼規(guī)范的 Class 文件,Spring AOP 使用了 AspectJ 。
Javassist 直接使用 Java 編碼的形式操作字節(jié)碼,簡(jiǎn)單易上手,性能高于反射,相比于 ASM 稍低。
Javassist 常用類
Javassist 抽象出一個(gè) ClassPool 對(duì)象來(lái)操作 Java 類,可以通過(guò) ClassPool.getDefault() 來(lái)獲取默認(rèn)的 ClassPool 。常用的對(duì)象:
CtClass:代表一個(gè) Class 的實(shí)例,可以通過(guò)類的全限定名來(lái)獲取 CtClass 對(duì)象,其中包含了對(duì) Class 的各種操作。
ClassPool:通過(guò) HashTable 保存了路徑下的 CtClass 信息,key為類的全限定名稱,value 為類名對(duì)應(yīng)的 CtClass 對(duì)象。
CtMethod、CtField:抽象出類的方法和屬性,可以用于定義或修改方法和字段。
Javassist 的使用
依賴
<dependency> <groupId>org.javassist</groupId> <artifactId>javassist</artifactId> <version>3.27.0-GA</version> </dependency>
代碼示例
// 獲取默認(rèn)類池
ClassPool classPool = ClassPool.getDefault();
// 1. 創(chuàng)建空類
CtClass ctClass = classPool.makeClass("com.aysaml.demo.javassist.User");
// 2. 創(chuàng)建 String 類型的 name 字段
CtField field = new CtField(classPool.get("java.lang.String"), "name", ctClass);
// 設(shè)置字段訪問(wèn)級(jí)別 private
field.setModifiers(Modifier.PRIVATE);
// 增加字段
ctClass.addField(field);
// 3. 增加 getter & setter 方法
ctClass.addMethod(CtNewMethod.getter("getName", field));
ctClass.addMethod(CtNewMethod.setter("setName", field));
// 4. 增加無(wú)參構(gòu)造方法:其中 $0 表示 this,$1 表示參數(shù)
CtConstructor noArgsCons = new CtConstructor(new CtClass[] {}, ctClass);
noArgsCons.setBody("{$0.name=\"mark\";}");
ctClass.addConstructor(noArgsCons);
// 5. 增加有參構(gòu)造方法
CtConstructor hasArgsCons =
new CtConstructor(new CtClass[] {classPool.get("java.lang.String")}, ctClass);
hasArgsCons.setBody("{$0.name=$1;}");
ctClass.addConstructor(hasArgsCons);
// 6. 創(chuàng)建方法
CtMethod method = new CtMethod(CtClass.voidType, "printName", new CtClass[] {}, ctClass);
method.setBody("{System.out.println($0.name);}");
ctClass.addMethod(method);
// 7. 生成類文件:可指定路徑,默認(rèn)為當(dāng)前項(xiàng)目根目錄
ctClass.writeFile();
// 8. 創(chuàng)建類實(shí)例
Object person = ctClass.toClass().newInstance();如何實(shí)現(xiàn)類似 AOP 的功能
javassist 對(duì)于編程化的操作字節(jié)碼是很簡(jiǎn)單易懂的,我們以在方法的開(kāi)頭結(jié)尾打印信息為例:
public class Cat {
/** 記錄喵喵喵的次數(shù) */
private int num;
public void miao() {
this.num++;
}
}我們要在 miao( ) 方法的前增加聲音輸出:
public static void main(String[] args) throws NotFoundException, CannotCompileException {
ClassPool classPool = ClassPool.getDefault();
// 獲取 Cat 類的 CtClass 對(duì)象
CtClass catClass = classPool.get("com.aysaml.demo.javassist.Cat");
// 獲取 miao( ) 方法
CtMethod method = catClass.getDeclaredMethod("miao");
method.insertBefore("System.out.println(\"miao~\");");
// 加載修改過(guò)的類,注意必須要保證調(diào)用前這個(gè)類沒(méi)有被加載過(guò)
catClass.toClass();
//測(cè)試
Cat cat = new Cat();
cat.miao();
}注意到,在使用 catClass.toClass() 加載被修改過(guò)的類時(shí),強(qiáng)調(diào)必須保證在調(diào)用前這個(gè)類沒(méi)有被加載過(guò),否則會(huì)報(bào) attempted duplicate class definition for name 異常。
我們知道一個(gè)類是不能被一個(gè)類加載器加載兩次的,所以為了解決這個(gè)問(wèn)題,需要制定一個(gè)沒(méi)有加載過(guò)該類的 Classloader,Javassist 提供了一個(gè) ClassLoader ,如下:
public class Cat {
/** 記錄喵喵喵的次數(shù) */
private int num;
public void miao() {
System.out.println("調(diào)用了 miao 方法");
this.num++;
}
public static void main(String[] args) throws Exception{
ClassPool classPool = ClassPool.getDefault();
// 獲取 Cat 類的 CtClass 對(duì)象
CtClass catClass = classPool.get("com.aysaml.demo.javassist.Cat");
// 獲取 miao( ) 方法
CtMethod method = catClass.getDeclaredMethod("miao");
method.insertBefore("System.out.println(\"miao~\");");
// 重新設(shè)置一個(gè) Classloader
Loader classLoader = new Loader(classPool);
Class clazz = classLoader.loadClass("com.aysaml.demo.javassist.Cat");
// 調(diào)用修改過(guò)的類的方法
clazz.getDeclaredMethod("miao").invoke(clazz.newInstance());
}
}執(zhí)行結(jié)果為:

到此這篇關(guān)于詳解Java中Javassist的使用的文章就介紹到這了,更多相關(guān)Java Javassist內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
Springboot項(xiàng)目刪除項(xiàng)目同步target文件問(wèn)題解決方案
這篇文章主要介紹了Springboot項(xiàng)目刪除項(xiàng)目同步target文件問(wèn)題解決方案,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下2020-12-12
spring security與corsFilter沖突的解決方案
這篇文章主要介紹了spring security與corsFilter沖突的解決方案,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2021-11-11
Maven之遠(yuǎn)程倉(cāng)庫(kù)的配置詳解
這篇文章主要介紹了Maven之遠(yuǎn)程倉(cāng)庫(kù)的配置詳解,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2020-09-09
Spring Security OAuth2 token權(quán)限隔離實(shí)例解析
這篇文章主要介紹了Spring Security OAuth2 token權(quán)限隔離實(shí)例解析,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下2019-11-11
Java中與數(shù)字相關(guān)的常用類的用法詳解
在我們的代碼中,經(jīng)常會(huì)遇到一些數(shù)字&數(shù)學(xué)問(wèn)題、隨機(jī)數(shù)問(wèn)題、日期問(wèn)題和系統(tǒng)設(shè)置問(wèn)題等,為了解決這些問(wèn)題,Java給我們提供了多個(gè)處理相關(guān)問(wèn)題的類,比如Number類、Math類、Random類等等,本篇文章我們先從Number數(shù)字類和Math數(shù)學(xué)類學(xué)起2023-05-05
Intellij Idea 多模塊Maven工程中模塊之間無(wú)法相互引用問(wèn)題
這篇文章主要介紹了Intellij Idea 多模塊Maven工程中模塊之間無(wú)法相互引用問(wèn)題,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2021-01-01

