一区二区三区在线-一区二区三区亚洲视频-一区二区三区亚洲-一区二区三区午夜-一区二区三区四区在线视频-一区二区三区四区在线免费观看

服務器之家:專注于服務器技術(shù)及軟件下載分享
分類導航

PHP教程|ASP.NET教程|Java教程|ASP教程|編程技術(shù)|正則表達式|C/C++|IOS|C#|Swift|Android|VB|R語言|JavaScript|易語言|vb.net|

服務器之家 - 編程語言 - Java教程 - Javassist如何操作Java 字節(jié)碼

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

2020-09-04 00:12碼戀 Java教程

這篇文章主要介紹了Javassist如何操作Java 字節(jié)碼,幫助大家更好的理解和學習Java,感興趣的朋友可以了解下

一、開篇

說起 AOP 小伙伴們肯定很熟悉,無論是 JDK 動態(tài)代理或者是 CGLIB 等,其底層都是通過操作 Java 字節(jié)碼來實現(xiàn)代理。常用的一些操作字節(jié)碼的技術(shù)有 ASM、AspectJ、Javassist 等。

ASM 其設計和實現(xiàn)是盡可能小而且快,更專注于性能。它在指令的層面來操作,所以使用它需要對 JVM 的指令有所了解,門檻較高,CGLIB 就使用了 ASM 技術(shù)。
AspectJ 擴展了 Java 語言,定義了一系列 AOP 語法,在 JVM 中運行需要使用特定的編譯器生成遵守 Java 字節(jié)碼規(guī)范的 Class 文件,Spring AOP 使用了 AspectJ 。
Javassist 直接使用 Java 編碼的形式操作字節(jié)碼,簡單易上手,性能高于反射,相比于 ASM 稍低。

二、Javassist 常用類

Javassist 抽象出一個 ClassPool 對象來操作 Java 類,可以通過 ClassPool.getDefault() 來獲取默認的 ClassPool 。常用的對象:

CtClass:代表一個 Class 的實例,可以通過類的全限定名來獲取 CtClass 對象,其中包含了對 Class 的各種操作。
ClassPool:通過 HashTable 保存了路徑下的 CtClass 信息,key為類的全限定名稱,value 為類名對應的 CtClass 對象。
CtMethod、CtField:抽象出類的方法和屬性,可以用于定義或修改方法和字段。

三、Javassist 的使用

1、依賴

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

2、代碼示例

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
// 獲取默認類池
  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);
  // 設置字段訪問級別 private
  field.setModifiers(Modifier.PRIVATE);
  // 增加字段
  ctClass.addField(field);
 
  // 3. 增加 getter & setter 方法
  ctClass.addMethod(CtNewMethod.getter("getName", field));
  ctClass.addMethod(CtNewMethod.setter("setName", field));
 
  // 4. 增加無參構(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. 生成類文件:可指定路徑,默認為當前項目根目錄
  ctClass.writeFile();
 
  // 8. 創(chuàng)建類實例
  Object person = ctClass.toClass().newInstance();

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

由上可見,Javassist 對于編程化的操作字節(jié)碼是很簡單易懂的,我們以在方法的開頭結(jié)尾打印信息為例:

?
1
2
3
4
5
6
7
8
9
public class Cat {
 
 /** 記錄喵喵喵的次數(shù) */
 private int num;
 
 public void miao() {
  this.num++;
 }
}

我們要在 miao( ) 方法的前增加聲音輸出:

?
1
2
3
4
5
6
7
8
9
10
11
12
13
public static void main(String[] args) throws NotFoundException, CannotCompileException {
  ClassPool classPool = ClassPool.getDefault();
  // 獲取 Cat 類的 CtClass 對象
  CtClass catClass = classPool.get("com.aysaml.demo.javassist.Cat");
  // 獲取 miao( ) 方法
  CtMethod method = catClass.getDeclaredMethod("miao");
  method.insertBefore("System.out.println(\"miao~\");");
  // 加載修改過的類,注意必須要保證調(diào)用前這個類沒有被加載過
  catClass.toClass();
  //測試
  Cat cat = new Cat();
  cat.miao();
 }

注意到,在使用 catClass.toClass() 加載被修改過的類時,強調(diào)必須保證在調(diào)用前這個類沒有被加載過,否則會報 attempted duplicate class definition for name 異常。

我們知道一個類是不能被一個類加載器加載兩次的,所以為了解決這個問題,需要制定一個沒有加載過該類的 Classloader,Javassist 提供了一個 ClassLoader ,如下:

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
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 對象
  CtClass catClass = classPool.get("com.aysaml.demo.javassist.Cat");
  // 獲取 miao( ) 方法
  CtMethod method = catClass.getDeclaredMethod("miao");
  method.insertBefore("System.out.println(\"miao~\");");
  // 重新設置一個 Classloader
  Loader classLoader = new Loader(classPool);
  Class clazz = classLoader.loadClass("com.aysaml.demo.javassist.Cat");
  // 調(diào)用修改過的類的方法
  clazz.getDeclaredMethod("miao").invoke(clazz.newInstance());
 }
}

執(zhí)行結(jié)果為:

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

四、結(jié)語

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

以上就是Javassist如何操作Java 字節(jié)碼的詳細內(nèi)容,更多關(guān)于Javassist 操作Java 字節(jié)碼的資料請關(guān)注服務器之家其它相關(guān)文章!

原文鏈接:https://www.tuicool.com/articles/QJBjmaR

延伸 · 閱讀

精彩推薦
主站蜘蛛池模板: 亚洲男人天堂网站 | 亚洲国产精品日韩高清秒播 | 给我视频免费看 | 精品女同一区二区三区免费站 | 亚洲激情一区 | 亚洲欧美视频在线播放 | 教室眠催白丝美女校花 | 无码爽死成人777在线观看网站 | 免费看黄色一级 | 黄网在线观看免费网站台湾swag | 色婷婷久久综合中文久久一本` | 亚洲AV无码A片在线观看蜜桃 | 日本三不卡| 欧美成人一区二区 | 日本大片免aaa费观看视频 | blacked亚裔videoshd| www.精品在线| 国产免费成人在线视频 | 三叶草私人研究所 | 国产japanese孕妇孕交 | 亚洲精品国产精品精 | 国产亚洲精品一区二区在线播放 | 91精品国产美女福到在线不卡 | 99精品热线在线观看免费视频 | 欧美性xxxxx 欧美性bbbbbxxxxxddd | 日本xnxnxnxnxn护士 | 国产中文在线 | 操久| bl高h荡肉古代np | 欧美日韩国产超高清免费看片 | 国产成人精品午夜视频' | 女bbbxxx毛片视频 | 问一问免费咨询 | 给我一个黄色网址 | 成人青青草| 国产日韩欧美综合在线 | 欧美三级一区 | 美女扒开肌肌让男人桶 | 息与子中文字幕完整在线 | 免费观看成年肉动漫网站 | 波多野结衣中文字幕在线 |