亚洲精品久久久中文字幕-亚洲精品久久片久久-亚洲精品久久青草-亚洲精品久久婷婷爱久久婷婷-亚洲精品久久午夜香蕉

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

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

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

一、開(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)是盡可能小而且快,更專(zhuān)注于性能。它在指令的層面來(lái)操作,所以使用它需要對(duì) JVM 的指令有所了解,門(mén)檻較高,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 常用類(lèi)

Javassist 抽象出一個(gè) ClassPool 對(duì)象來(lái)操作 Java 類(lèi),可以通過(guò) ClassPool.getDefault() 來(lái)獲取默認(rèn)的 ClassPool 。常用的對(duì)象:

CtClass:代表一個(gè) Class 的實(shí)例,可以通過(guò)類(lèi)的全限定名來(lái)獲取 CtClass 對(duì)象,其中包含了對(duì) Class 的各種操作。ClassPool:通過(guò) HashTable 保存了路徑下的 CtClass 信息,key為類(lèi)的全限定名稱(chēng),value 為類(lèi)名對(duì)應(yīng)的 CtClass 對(duì)象。CtMethod、CtField:抽象出類(lèi)的方法和屬性,可以用于定義或修改方法和字段。

三、Javassist 的使用

1、依賴(lài)

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

2、代碼示例

// 獲取默認(rèn)類(lèi)池 ClassPool classPool = ClassPool.getDefault(); // 1. 創(chuàng)建空類(lèi) CtClass ctClass = classPool.makeClass('com.aysaml.demo.javassist.User'); // 2. 創(chuàng)建 String 類(lèi)型的 name 字段 CtField field = new CtField(classPool.get('java.lang.String'), 'name', ctClass); // 設(shè)置字段訪(fǎng)問(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. 生成類(lèi)文件:可指定路徑,默認(rèn)為當(dāng)前項(xiàng)目根目錄 ctClass.writeFile(); // 8. 創(chuàng)建類(lèi)實(shí)例 Object person = ctClass.toClass().newInstance();

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

由上可見(jiàn),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 類(lèi)的 CtClass 對(duì)象 CtClass catClass = classPool.get('com.aysaml.demo.javassist.Cat'); // 獲取 miao( ) 方法 CtMethod method = catClass.getDeclaredMethod('miao'); method.insertBefore('System.out.println('miao~');'); // 加載修改過(guò)的類(lèi),注意必須要保證調(diào)用前這個(gè)類(lèi)沒(méi)有被加載過(guò) catClass.toClass(); //測(cè)試 Cat cat = new Cat(); cat.miao(); }

注意到,在使用 catClass.toClass() 加載被修改過(guò)的類(lèi)時(shí),強(qiáng)調(diào)必須保證在調(diào)用前這個(gè)類(lèi)沒(méi)有被加載過(guò),否則會(huì)報(bào) attempted duplicate class definition for name 異常。

我們知道一個(gè)類(lèi)是不能被一個(gè)類(lèi)加載器加載兩次的,所以為了解決這個(gè)問(wèn)題,需要制定一個(gè)沒(méi)有加載過(guò)該類(lèi)的 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 類(lèi)的 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ò)的類(lèi)的方法 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)文章:
主站蜘蛛池模板: 九九热在线视频免费观看 | 亚洲黄色片 | 欧美日韩国产在线 | 色婷婷久久免费网站 | 国产高清视频在线播放 | 日本特黄特色免费大片 | 精品久久久久久亚洲 | 亚洲欧美综合色区小说 | 黄色一级片在线免费观看 | 国产视频自拍一区 | 久久成人免费视频 | 欧美黄色一级片免费看 | 12306播播影院午夜爱我影院 | 中日韩欧美视频 | 一本伊大人香蕉在线观看 | 男人和女人做a免费视频 | 嫩草影院在线入口 | 大乳女人做受视频免费观看 | 国产黄色高清视频 | 亚洲香蕉综合在人在线时看 | 韩日在线视频 | 精品国产日韩亚洲一区二区 | 婷婷国产成人久久精品激情 | 大学生久久香蕉国产线观看 | 欧美视频免费一区二区三区 | 亚洲精品午夜国产va久久成人 | 国产一级特黄aa级特黄裸毛片 | 国产永久免费视频 | 97视频免费在线观看 | 国产视频一二三区 | 高清波多野结衣一区二区三区 | 爱综合网 | 欧美三级在线免费观看 | 鲁丝片一区二区三区 | 国内免费视频成人精品 | 美女被免费网站视频九色 | 国产大片喷水在线在线视频 | 激情亚洲综合网 | 96免费精品视频在线 | 国产精品久久久久影院免费 | 国产自啪啪 |