cglib FastClass機制

来源:https://www.cnblogs.com/jtea/p/18067745
-Advertisement-
Play Games

前言 關於動態代理的一些知識,以及cglib與jdk動態代理的區別,在這一篇已經介紹過,不熟悉的可以先看下。 本篇我們來學習一下cglib的FastClass機制,這是cglib與jdk動態代理的一個主要區別,也是一個面試考點。 我們知道jdk動態代理是使用InvocationHandler介面,在 ...


前言

關於動態代理的一些知識,以及cglib與jdk動態代理的區別,在這一篇已經介紹過,不熟悉的可以先看下。
本篇我們來學習一下cglib的FastClass機制,這是cglib與jdk動態代理的一個主要區別,也是一個面試考點。
我們知道jdk動態代理是使用InvocationHandler介面,在invoke方法內,可以使用Method方法對象進行反射調用,反射的一個最大問題是性能較低,cglib就是通過使用FastClass來優化反射調用,提升性能,接下來我們就看下它是如何實現的。

示例

我們先寫一個hello world,讓代碼跑起來。如下:

public class HelloWorld {

	public void print() {
		System.out.println("hello world");
	}
}

public class HelloWorldInterceptor implements MethodInterceptor {
	public Object intercept(Object o, Method method, Object[] objects, MethodProxy methodProxy) throws Throwable {
		System.out.println("before hello world");
		methodProxy.invokeSuper(o, objects);
		System.out.println("after hello world");
		return null;
	}
}

非常簡單,就是使用MethodInterceptor在HelloWorld類print方法前後列印一句話,模擬對一個方法前後織入自定義邏輯。
接著使用cglib Enhancer類,創建動態代理對象,設置MethodInterceptor,調用方法。
為了方便觀察源碼,我們將cglib生成的動態代理類保存下來。


//將生成的動態代理類保存下來
System.setProperty(DebuggingClassWriter.DEBUG_LOCATION_PROPERTY, "D:\\");

Enhancer enhancer = new Enhancer();
enhancer.setSuperclass(HelloWorld.class);
enhancer.setCallback(new HelloWorldInterceptor());

HelloWorld target = (HelloWorld) enhancer.create();
target.print();

輸出

before hello world
hello world
after hello world

FastClass機制

我們知道cglib是通過繼承實現的,動態代理類會繼承被代理類,並重寫它的方法,所以它不需要像jdk動態代理一樣要求被代理對象有實現介面,因此比較靈活。
既然是通過繼承實現的,那應該生成一個類就可以了,但是通過上面的路徑觀察,可以看到生成了3個文件,其中兩個帶有FastClass關鍵字。
這三個類分別是:動態代理類,動態代理類的FastClass,被代理對象的FastClass,從名稱上也可以看出它們的關係。

其中動態代理類繼承了被代理類,並重寫了父類的所有方法,包括父類的父類的方法,包括Object類的equals方法和toString方法等。

public class HelloWorld$$EnhancerByCGLIB$$49f9f9c8 extends HelloWorld implements Factory {
}

這裡我們只關註print方法,如下:

第一個直接調用父類方法,也就是被代理對象的方法;第二個會先判斷有沒有攔截器,如果沒有也是直接調用父類方法,否則調用MethodInterceptor的intercept方法,對於我們這裡就是HelloWorldInterceptor。
看下intercept的幾個參數分別是什麼,這幾個參數的初始化在動態代理類的靜態代碼塊中都可以找到。
第1個表示動態代理對象。
第2個是被代理對象方法的Method,就是HelloWorld.print。
第3個表示方法參數。
第4個是MethodProxy對象,通過名字我們可以知道它是方法的代理,每一個方法都會有一個對應的MethodProxy,它包含被代理對象、代理對象、以及對應的方法元信息。

這裡我們重點關註MethodProxy,它的初始化如下:

CGLIB$print$0$Proxy = MethodProxy.create(var1, var0, "()V", "print", "CGLIB$print$0");       

第1個參數表示被代理對象的Class。
第2個參數表示動態代理對象的Class。
第3個參數是方法的返回值。
第4個參數表示被代理對象的方法名稱。
第5個參數表示對應動態代理對象的方法名稱。

MethodProxy對象創建好後,我們上面就是通過它進行調用的

methodProxy.invokeSuper(o, objects);

invokeSuper主要源碼如下:

public Object invokeSuper(Object obj, Object[] args) throws Throwable {
    init();
    FastClassInfo fci = fastClassInfo;
    return fci.f2.invoke(fci.i2, obj, args);
}

private void init()
{
    if (fastClassInfo == null)
    {
        synchronized (initLock)
        {
            if (fastClassInfo == null)
            {
                CreateInfo ci = createInfo;

                FastClassInfo fci = new FastClassInfo();
                fci.f1 = helper(ci, ci.c1); //被代理對象的FastClass
                fci.f2 = helper(ci, ci.c2); //動態代理對象的FastClass
                fci.i1 = fci.f1.getIndex(sig1); //被代理對象方法的索引下標
                fci.i2 = fci.f2.getIndex(sig2); //動態代理對象方法的索引下標,這裡是:CGLIB$print$0 
                fastClassInfo = fci;
                createInfo = null;
            }
        }
    }
}

init方法使用加鎖+雙檢查的方式,只會初始化一次fastClassInfo變數,它用volatile關鍵字進行修飾,這裡涉及到java位元組碼重排問題,具體可以參考我們之前的分析:happend before原則

接著回到invokeSuper方法,fci.f2.invoke(fci.i2, obj, args); 實際就是調用動態代理對象的FastClass的invoke方法,並把要調用方法的索引下標i2傳過去。
至於方法的索引下標是怎麼找到的,可以看動態代理對象的FastClass的getIndex方法,其實就是通過方法的名稱、參數個數、參數類型,完全匹配,點到源碼文件可以看到有大量的switch分支判斷。
這裡我們可以看到print方法的索引下標就是18。

public int getIndex(String var1, Class[] var2) {
    switch (var1.hashCode()) {
        case -1295482945:
            if (var1.equals("equals")) {
                switch (var2.length) {
                    case 1:
                        if (var2[0].getName().equals("java.lang.Object")) {
                            return 0;
                        }
                }
            }
        break;
        case 770871766:
            if (var1.equals("CGLIB$print$0")) {
                switch (var2.length) {
                    case 0:
                        return 18;
                }
            }
        break;
    }
}
 public Object invoke(int var1, Object var2, Object[] var3) throws InvocationTargetException {
    HelloWorld..EnhancerByCGLIB..49f9f9c8 var10000 = (HelloWorld..EnhancerByCGLIB..49f9f9c8)var2;
    int var10001 = var1;

    //...
    switch (var10001) {                
        //...
        case 18:
            var10000.CGLIB$print$0();
            return null;
    }
 }    

可以看到最終調用到動態代理類的CGLIB$print$0方法,也就是:

    final void CGLIB$print$0() {
        super.print();
    }

最終調用的就是父類的方法。我們畫張圖總結一下,有興趣的同學跟著圖和代碼邏輯應該可以快速理解。

總結

經過上面的分析,我們可以看到cglib在整個調用過程並沒有用到反射,而是使用FastClass對每個方法進行索引,通過方法名稱,參數長度,參數類型就可以找到具體的方法,因此性能較好。但也有缺點,首次調用需要生成3個類,會比較慢。在我們實際開發中,特別是一些框架開發,如果有類似的場景也可以藉助FastClass對反射進行優化,如:

MyClass cs = new MyCase();
FastClass fastClass = FastClass.create(Case.class);
int index = fastClass.getIndex("test", new Class[]{Integer.class});
Object invoke = fastClass.invoke(index, cs, new Object[1]);

另外MethodProxy還有一個invoke方法,如果我們換一下調用這個方法會發生?留給大家自己嘗試。

methodProxy.invokeSuper(o, objects);
//換成 methodProxy.invoke(o, objects);

更多分享,歡迎關註我的github:https://github.com/jmilktea/jtea


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

-Advertisement-
Play Games
更多相關文章
  • Java Iterator Iterator 介面提供了一種迭代集合的方法,即順序訪問集合中的每個元素。它支持 hasNext() 和 next() 方法,用於檢查是否存在下一個元素以及獲取下一個元素。 獲取 Iterator 可以使用集合的 iterator() 方法獲取 Iterator 實例: ...
  • 摘要: 銀行卡歸屬地查詢介面是一種高效的方式,通過銀行卡號查詢銀行名稱、卡種、卡品牌以及發卡省份和城市等信息。本文將詳細介紹如何使用該介面,並附帶代碼說明。同時,也介紹了介面的特點和適用範圍,讓讀者能夠充分瞭解和運用該介面,方便快捷地獲取銀行卡發卡行所在地信息。 一、介面簡介 銀行卡歸屬地查詢介面是 ...
  • 1. 本篇文章目標 將下麵的excel中的寄存器表單讀入並構建一個字典 2. openpyxl的各種基本使用方法 2.1 打開工作簿 wb = openpyxl.load_workbook('test_workbook.xlsx') 2.2 獲取工作簿中工作表名字並得到工作表 ws = wb[wb. ...
  • 拓展閱讀 linux Shell 命令行-00-intro 入門介紹 linux Shell 命令行-02-var 變數 linux Shell 命令行-03-array 數組 linux Shell 命令行-04-operator 操作符 linux Shell 命令行-05-test 驗證是否符 ...
  • 在之前的多線程系列文章中,我們陸陸續續的介紹了Thread線程類相關的知識和用法,其實在Thread類上還有一層ThreadGroup類,也就是線程組。 ...
  • 2024年3月4日,官方宣佈推出 Claude 3 模型系列,它在廣泛的認知任務中樹立了新的行業基準。該系列包括三個按能力遞增排序的最先進模型:Claude 3 Haiku、Claude 3 Sonnet 和 Claude 3 Opus。每個後續模型都提供越來越強大的性能,允許用戶為其特定應用選擇智 ...
  • 大家好,我是你們的老伙計秀才!今天帶來的是[深入淺出Java多線程]系列的第十一篇內容:AQS(*AbstractQueuedSynchronizer*)。大家覺得有用請點贊,喜歡請關註!秀才在此謝過大家了!!! ...
  • Qt 是一個跨平臺C++圖形界面開發庫,利用Qt可以快速開發跨平臺窗體應用程式,在Qt中我們可以通過拖拽的方式將不同組件放到指定的位置,實現圖形化開發極大的方便了開發效率,本章將重點介紹如何運用`QNetworkAccessManager`組件實現Web網頁訪問。QNetworkAccessMana... ...
一周排行
    -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.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...