日韩性视频-久久久蜜桃-www中文字幕-在线中文字幕av-亚洲欧美一区二区三区四区-撸久久-香蕉视频一区-久久无码精品丰满人妻-国产高潮av-激情福利社-日韩av网址大全-国产精品久久999-日本五十路在线-性欧美在线-久久99精品波多结衣一区-男女午夜免费视频-黑人极品ⅴideos精品欧美棵-人人妻人人澡人人爽精品欧美一区-日韩一区在线看-欧美a级在线免费观看

歡迎訪問 生活随笔!

生活随笔

當前位置: 首頁 > 编程资源 > 编程问答 >内容正文

编程问答

CGLib 调用API 及原理分析

發布時間:2024/4/13 编程问答 22 豆豆
生活随笔 收集整理的這篇文章主要介紹了 CGLib 调用API 及原理分析 小編覺得挺不錯的,現在分享給大家,幫大家做個參考.

簡單看一下CGLib 代理的使用,還是以媒婆為例,創建CglibMeipo 類:

public class CGlibMeipo implements MethodInterceptor {public Object getInstance(Class<?> clazz) throws Exception{//相當于Proxy,代理的工具類Enhancer enhancer = new Enhancer();enhancer.setSuperclass(clazz);enhancer.setCallback(this);return enhancer.create();}public Object intercept(Object o, Method method, Object[] objects, MethodProxy methodProxy) throws Throwable {before();Object obj = methodProxy.invokeSuper(o,objects);after();return obj;}private void before(){System.out.println("我是媒婆,我要給你找對象,現在已經確認你的需求");System.out.println("開始物色");}private void after(){System.out.println("OK的話,準備辦事");} }

創建單身客戶Customer 類:

public class Customer {public void findLove(){System.out.println("兒子要求:膚白貌美大長腿");} }

有個小細節,CGLib 代理的目標對象不需要實現任何接口,它是通過動態繼承目標對象實現的動態代理。來看測試代碼:

public class CglibTest {public static void main(String[] args) {try {//JDK是采用讀取接口的信息//CGLib覆蓋父類方法//目的:都是生成一個新的類,去實現增強代碼邏輯的功能//JDK Proxy 對于用戶而言,必須要有一個接口實現,目標類相對來說復雜//CGLib 可以代理任意一個普通的類,沒有任何要求//CGLib 生成代理邏輯更復雜,效率,調用效率更高,生成一個包含了所有的邏輯的FastClass,不再需要反射調用//JDK Proxy生成代理的邏輯簡單,執行效率相對要低,每次都要反射動態調用//CGLib 有個坑,CGLib不能代理final的方法System.setProperty(DebuggingClassWriter.DEBUG_LOCATION_PROPERTY,"E://cglib_proxy_classes");Customer obj = (Customer) new CGlibMeipo().getInstance(Customer.class);System.out.println(obj);obj.findLove();} catch (Exception e) {e.printStackTrace();}} }

CGLib 的實現原理又是怎樣的呢?我們可以在測試代碼中加上一句代碼,將CGLib 代理后的class 寫入到磁盤,然后,我們再反編譯一探究竟,重寫了Customer 類的所有方法。我們通過代理類的源碼可以看到,代理類會獲得所有在父類繼承來的方法, 并且會有MethodProxy 與之對應, 比如MethodCGLIB$findLove$0$Method、MethodProxy CGLIB$findLove$0$Proxy;這些方法在代理類的findLove()中都有調用。

調用過程: 代理對象調用this.findLove() 方法-> 調用攔截器->methodProxy.invokeSuper->CGLIB$findLove$0->被代理對象findLove()方法。此時,我們發現攔截器MethodInterceptor 中就是由MethodProxy 的invokeSuper方法調用代理方法的,MethodProxy 非常關鍵,我們分析一下它具體做了什么。

public class MethodProxy {private Signature sig1;private Signature sig2;private MethodProxy.CreateInfo createInfo;private final Object initLock = new Object();private volatile MethodProxy.FastClassInfo fastClassInfo;public static MethodProxy create(Class c1, Class c2, String desc, String name1, String name2) {MethodProxy proxy = new MethodProxy();proxy.sig1 = new Signature(name1, desc);proxy.sig2 = new Signature(name2, desc);proxy.createInfo = new MethodProxy.CreateInfo(c1, c2);return proxy;} }

CGLib 動態代理執行代理方法效率之所以比JDK 的高是因為Cglib 采用了FastClass 機制,它的原理簡單來說就是:為代理類和被代理類各生成一個Class,這個Class 會為代理類或被代理類的方法分配一個index(int 類型)。這個index 當做一個入參,FastClass就可以直接定位要調用的方法直接進行調用,這樣省去了反射調用,所以調用效率比JDK動態代理通過反射調用高。下面我們反編譯一個FastClass 看看:

?

總結

以上是生活随笔為你收集整理的CGLib 调用API 及原理分析的全部內容,希望文章能夠幫你解決所遇到的問題。

如果覺得生活随笔網站內容還不錯,歡迎將生活随笔推薦給好友。