Java虛擬機知識點【位元組碼】

来源:https://www.cnblogs.com/bigshark/archive/2019/07/24/11235459.html
-Advertisement-
Play Games

位元組碼指令   Java虛擬機的位元組碼指令由一個位元組長度,代表著某種特定操作含義的操作碼以及跟隨其後的零至多個代表此操作所需參數的操作數所構成的。如果忽略異常,JVM的解釋器通過下麵的偽代碼可有效工作: 操作位元組碼   可以利用開源庫直接操作位元組碼,如CGLi ...


位元組碼指令

  Java虛擬機的位元組碼指令由一個位元組長度,代表著某種特定操作含義的操作碼以及跟隨其後的零至多個代表此操作所需參數的操作數所構成的。如果忽略異常,JVM的解釋器通過下麵的偽代碼可有效工作:

do {
  自動計算PC寄存器以及從PC寄存器的位置取出操作碼;
if (存在操作數) 取出操作數;
  執行操作碼所定義的操作;
} while (處理下一次迴圈);

操作位元組碼

  可以利用開源庫直接操作位元組碼,如CGLib、ASM、Javassist等,他們可以在程式運行時,動態地創建位元組碼類或者編輯存在的位元組碼類。其中,CGLib是基於ASM實現的,是一個高效高性能的生成庫;而ASM是一個輕量級的類庫,但需要涉及到JVM的操作和指令;相比而言,Javassist要簡單的多,完全是基於Java的API,但其性能相比前二者要差一些。

使用CGLib實現動態代理

  Java 編譯完後不會立即生成代理類,而是在運行時動態生成代理類位元組碼,並載入到記憶體中。通過實現JDK的介面 InvocationHandler 就可以來實現動態代理。

// 示例:JDK實現動態代理
public class ProxyHandler implements InvocationHandler {
  @Override
  public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
    System.out.println("Start");
    Object o = method.invoke(this, args);
    System.out.println("End");
    return o;
  }
}

  使用JDK實現動態代理的類必須要實現一個介面,在實際開發中有一定的局限性,反射的效率也並不是很高,因此可以利用操作位元組碼技術來實現動態代理。流行的開發框架 Spring 則同時實現了這兩種方式,可以在實際開發中選擇基於JDK的動態代理,或者基於CGLib的動態代理。

CGLib 的 github地址

通過CGLib來實現動態代理需要引入CGLib和asm的依賴包

<!-- cglib -->
<dependency>
  <groupId>cglib</groupId>
  <artifactId>cglib</artifactId>
  <version>3.2.12</version>
</dependency>
<!-- asm -->
<dependency>
  <groupId>asm</groupId>
  <artifactId>asm</artifactId>
  <version>3.3.1</version>
</dependency>

需要被代理的類

// 需要被代理的類
public class HelloCGLib {

  public HelloCGLib() {
    System.out.println("HelloCGLib構造器");
  }

  public void sayHello(String name) {
    System.out.println("HelloCGLib:" + name);
  }
}

實現CGLib的方法攔截器

// 實現CGLib的方法攔截器
public class ProxyInterceptor implements MethodInterceptor {

  /**
   * @param o           cglib生成的代理對象
   * @param method      被代理對象方法
   * @param objects     方法入參
   * @param methodProxy 代理方法
   */
  @Override
  public Object intercept(Object o, Method method, Object[] objects, MethodProxy methodProxy) throws Throwable {
    System.out.println("======前置通知======");
    Object object = methodProxy.invokeSuper(o, objects);
    System.out.println("======後者通知======");
    return object;
  }
}

測試類

public class Test {
  public static void main(String[] args) {
    // 代理類class文件存入本地磁碟方便我們反編譯查看源碼
    System.setProperty(DebuggingClassWriter.DEBUG_LOCATION_PROPERTY, "F:\\");
    // CGLIB 增強類
    Enhancer enhancer = new Enhancer();
    // 設置增強類對象的父類
    enhancer.setSuperclass(HelloCGLib.class);
    // 設置增強類的回調類
    enhancer.setCallback(new ProxyInterceptor());
    // 創建代理對象
    HelloCGLib proxy = (HelloCGLib) enhancer.create();
    // 通過代理對象調用目標方法
    proxy.sayHello("hehe");
  }
}

運行結果

HelloCGLib構造器
======前置通知======
HelloCGLib:hehe
======後者通知======

F 盤下生成的代理類

代理類

由於CGlib是通過對需要增強的類生成一個子類,並覆蓋其中的方法來實現動態代理的,所以CGlib可以為無介面的類直接做代理,但是不能為final類做代理。

參考資料:《深入理解Java虛擬機(第二版)》《Java虛擬機規範(Java SE 8版)》


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • docker exec -it imageId redis-cli docker container ls -a docker rm containerId ...
  • 我們稱以存在的用來派生新類的類為基類,又稱做父類,超類。由已存在的類派生出的新類稱為派生類,又稱為子類。從一個基類派生的繼承稱單繼承,從多個基類派生的繼承稱為多繼承。也就是說:一個類只能直接從一個類中繼承數據。 那麼,怎樣實現繼承呢?很簡單,我們只需要按照創建類的方法後跟一個單詞extends 父類 ...
  • 我原本以為在對鄉村的一切事物都很容易,但是每次我一回頭就有更多變更的請求紛至沓來。我快崩潰了! 當小編讀到這段話的時候,甚是感慨,設計模式的世界真的不是那麼容易弄懂的,好在我還在學習。 基本常識:策略模式和狀態模式是雙胞胎,在出生時才分開。 策略模式是圍繞可以互換的演算法來創建成功業務的。然而,狀態走 ...
  • 本書特色 本書特色 為了能讓更多的編程小白輕鬆地入門編程,我把高效學習法結合 Python 中的核心知識,寫成了這本書。隨意翻上幾頁,你就會發現這本書和其他編程書的不同,其中有大量的視覺化場景幫你理解概念,穿插了若幹有趣的小項目,最重要的是,這本書是為零基礎小白而設計的。 下載地址 為了能讓更多的編 ...
  • 五、PHP綜合應用 ftp、ssh、http、telnet、https ftp:File Transfer Protocol,文件傳輸協議,是應用層的協議,它基於傳輸層,為用戶服務,它們負責進行文件的傳輸,其預設埠是21。 ssh:Secure Shell,安全外殼協議,建立在應用層和傳輸層基礎上 ...
  • 基礎題 一、String,StringBuffer, StringBuilder 的區別是什麼?String為什麼是不可變的?1. String是字元串常量,StringBuffer和StringBuilder是字元串變數。StringBuffer是線程安全的,StringBuilder是非線程安全 ...
  • 一、高德軟體有限公司python試題及答案 1. 在python中, list, tuple, dict, set有什麼區別, 主要應用在什麼樣的場景? 定義: list: 鏈表, 有序的項目, 通過索引進行查找, 使用方括弧"[]"; tuple: 元組, 元組將多樣的對象集合到一起, 不能修改, ...
  • 本章內容主要分享多個module中的實體類集合生成到一個jar包中,並且發佈到遠程庫;這裡採用maven-assembly-plugin插件的功能來操作打包,內容不長卻貼近實戰切值得擁有,主要節點內容如: 多個module實體類集合打jar包 jar包打入本地庫 jar包上傳至遠程庫 多個modul ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...