久久福利_99r_国产日韩在线视频_直接看av的网站_中文欧美日韩_久久一

您的位置:首頁(yè)技術(shù)文章
文章詳情頁(yè)

Javassist如何操作Java 字節(jié)碼

瀏覽:3日期:2022-08-25 15:44:26

一、開篇

說(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 的使用

1、依賴

<dependency> <groupId>org.javassist</groupId> <artifactId>javassist</artifactId> <version>3.27.0-GA</version></dependency>

2、代碼示例

// 獲取默認(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();

3、如何實(shí)現(xiàn)類似 AOP 的功能

由上可見(jiàn),Javassist 對(duì)于編程化的操作字節(jié)碼是很簡(jiǎn)單易懂的,我們以在方法的開頭結(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é)果為:

Javassist如何操作Java 字節(jié)碼

四、結(jié)語(yǔ)

關(guān)于 Javassist 暫時(shí)就說(shuō)這么多了,更多使用方法參考官方 github wiki :

以上就是Javassist如何操作Java 字節(jié)碼的詳細(xì)內(nèi)容,更多關(guān)于Javassist 操作Java 字節(jié)碼的資料請(qǐng)關(guān)注好吧啦網(wǎng)其它相關(guān)文章!

標(biāo)簽: Java
相關(guān)文章:
主站蜘蛛池模板: 日韩欧美在线免费观看 | 天天干天天操 | 亚洲男人天堂2023 | 亚洲精品aaa| 精品成人免费一区二区在线播放 | 国产日韩一区二区 | 日本一区二区三区免费观看 | 97国产精品视频 | 午夜精品久久久久久久男人的天堂 | 91久久久久久久久 | av午夜电影 | 精品亚洲永久免费精品 | 欧美性影院 | 久久国产精品视频 | 日韩在线视频免费 | 国产视频欧美 | www中文字幕| 欧美一区二区三区精品 | 久久久国产一区二区 | 亚洲国产精品成人 | 一区二区免费视频 | av中文字幕在线播放 | 天天干干| 国产激情精品视频 | 国产一二在线 | 欧美日韩视频在线观看免费 | 精品毛片| 一级免费av| 91中文字幕在线观看 | 免费看国产片在线观看 | 亚洲一区二区三区高清 | www.一区| 在线观看国产 | 久久人人爽人人爽人人片亚洲 | 美女国产精品 | 91av久久 | 午夜亚洲福利 | 久久久国产精品入口麻豆 | 一级做a爰片性色毛片2021 | 第一色站 | 日韩欧美一级精品久久 |