欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

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

 更新時間:2020年09月03日 10:30:29   作者:碼戀  
這篇文章主要介紹了Javassist如何操作Java 字節(jié)碼,幫助大家更好的理解和學(xué)習(xí)Java,感興趣的朋友可以了解下

一、開篇

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

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

二、Javassist 常用類

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

CtClass:代表一個 Class 的實例,可以通過類的全限定名來獲取 CtClass 對象,其中包含了對 Class 的各種操作。
ClassPool:通過 HashTable 保存了路徑下的 CtClass 信息,key為類的全限定名稱,value 為類名對應(yīng)的 CtClass 對象。
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è)置字段訪問級別 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. 生成類文件:可指定路徑,默認(rèn)為當(dāng)前項目根目錄
  ctClass.writeFile();

  // 8. 創(chuàng)建類實例
  Object person = ctClass.toClass().newInstance();

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

由上可見,Javassist 對于編程化的操作字節(jié)碼是很簡單易懂的,我們以在方法的開頭結(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 對象
  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() 加載被修改過的類時,強(qiáng)調(diào)必須保證在調(diào)用前這個類沒有被加載過,否則會報 attempted duplicate class definition for name 異常。

我們知道一個類是不能被一個類加載器加載兩次的,所以為了解決這個問題,需要制定一個沒有加載過該類的 Classloader,Javassist 提供了一個 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 對象
  CtClass catClass = classPool.get("com.aysaml.demo.javassist.Cat");
  // 獲取 miao( ) 方法
  CtMethod method = catClass.getDeclaredMethod("miao");
  method.insertBefore("System.out.println(\"miao~\");");
  // 重新設(shè)置一個 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é)果為:

四、結(jié)語

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

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

相關(guān)文章

  • 解讀springboot配置mybatis的sql執(zhí)行超時時間(mysql)

    解讀springboot配置mybatis的sql執(zhí)行超時時間(mysql)

    這篇文章主要介紹了解讀springboot配置mybatis的sql執(zhí)行超時時間(mysql),具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教
    2023-01-01
  • java項目怎么集成stable diffusion圖文生成算法

    java項目怎么集成stable diffusion圖文生成算法

    在開發(fā)Java項目過程中,我們經(jīng)常需要使用消息傳遞來實現(xiàn)不同組件之間的通信,Stable Diffusion是一種基于消息傳遞的實時通信解決方案,使用Java調(diào)用外部服務(wù)(如Python腳本或API服務(wù)),這些服務(wù)運(yùn)行Stable Diffusion模型,本文將介紹如何將Stable Diffusion集成到Java項目
    2024-07-07
  • 從?PageHelper?到?MyBatis?Plugin執(zhí)行概要及實現(xiàn)原理

    從?PageHelper?到?MyBatis?Plugin執(zhí)行概要及實現(xiàn)原理

    這篇文章主要為大家介紹了從?PageHelper?到?MyBatis?Plugin執(zhí)行概要及實現(xiàn)原理,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪
    2022-09-09
  • Spring Boot之@Async異步線程池示例詳解

    Spring Boot之@Async異步線程池示例詳解

    在Spring Boot中,我們只需要通過使用@Async注解就能簡單的將原來的同步函數(shù)變?yōu)楫惒胶瘮?shù),下面這篇文章主要給大家介紹了關(guān)于Spring Boot之@Async異步線程池的相關(guān)資料,需要的朋友可以參考下
    2021-09-09
  • java Spring松耦合高效應(yīng)用簡單實例分析

    java Spring松耦合高效應(yīng)用簡單實例分析

    在Java項目,龐大的對象依賴關(guān)系將一直緊密耦合引起對象難以管理或修改。在這種情況下,可以使用Spring框架作為一個核心模塊輕松高效地管理所有的對象依賴。本文章向大家介紹Spring松耦合的實例,需要的朋友可以參考一下。
    2016-12-12
  • MybatisPlus為何可以不用@MapperScan詳解

    MybatisPlus為何可以不用@MapperScan詳解

    這篇文章主要給大家介紹了關(guān)于MybatisPlus為何可以不用@MapperScan的相關(guān)資料,文中通過圖文介紹的非常詳細(xì),對大家學(xué)習(xí)或者使用MybatisPlus具有一定的參考學(xué)習(xí)價值,需要的朋友可以參考下
    2023-04-04
  • Java中List集合的常用方法詳解

    Java中List集合的常用方法詳解

    本篇文章給大家?guī)淼膬?nèi)容是關(guān)于Java中List集合的常用方法詳解,有一定的參考價值,有需要的朋友可以參考一下,希望對你有所幫助。下面我們就來學(xué)習(xí)一下吧
    2021-11-11
  • Java集合框架之Map詳解

    Java集合框架之Map詳解

    這篇文章主要為大家詳細(xì)介紹了Java集合框架之Map,文中示例代碼介紹的非常詳細(xì),具有一定的參考價值,感興趣的小伙伴們可以參考一下,希望能夠給你帶來幫助
    2022-03-03
  • 解決Spring或SpringBoot開啟事務(wù)以后無法返回自增主鍵的問題

    解決Spring或SpringBoot開啟事務(wù)以后無法返回自增主鍵的問題

    這篇文章主要介紹了解決Spring或SpringBoot開啟事務(wù)以后無法返回自增主鍵的問題,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教
    2021-07-07
  • Java super關(guān)鍵字的使用方法詳解

    Java super關(guān)鍵字的使用方法詳解

    這篇文章主要介紹了Java super關(guān)鍵字的使用方法詳解的相關(guān)資料,希望通過本文能幫助到大家,讓大家對super關(guān)鍵字徹底掌握,需要的朋友可以參考下
    2017-10-10

最新評論