您好,登錄后才能下訂單哦!
JDK 動態代理使用起來非常簡單,但是它也有一定的局限性,這是因為 JDK 動態代理必須要實現一個或多個接口,如果不希望實現接口,則可以使用 CGLIB 代理。
CGLIB(Code Generation Library)是一個高性能開源的代碼生成包,它被許多 AOP 框架所使用,其底層是通過使用一個小而快的字節碼處理框架 ASM(Java 字節碼操控框架)轉換字節碼并生成新的類。因此 CGLIB 要依賴于 ASM 的包,解壓 Spring 的核心包 spring-core-3.2.2.RELEASE.jar,文件目錄如圖 1 所示。
圖 1 spring-core-3.2.2.RELEASE.jar文件
在圖 1 中可以看出,解壓的核心包中包含 cglib 和 asm,也就是說 Spring3.2.13 版本的核心包已經集成了 CGLIB 所需要的包,所以在開發中不需要另外導入 ASM 的 JAR 包了。下面通過案例演示實現 CGLIB 的代理過程。
1. 創建目標類 GoodsDao
在 com.mengma.dao 包下創建目標類 GoodsDao,在類中定義增、刪、改、查方法,并在每個方法編寫輸出語句,如下所示。
package com.mengma.dao; public class GoodsDao { public void add() { System.out.println("添加商品..."); } public void update() { System.out.println("修改商品..."); } public void delete() { System.out.println("刪除商品..."); } public void find() { System.out.println("修改商品..."); } }
2. 創建代理類 MyBeanFactory
在 src 目錄下創建一個名為 com.mengma.cglib 的包,該包下創建類 MyBeanFactory,如下所示。
package com.mengma.cglib; import java.lang.reflect.Method; import org.springframework.cglib.proxy.Enhancer; import org.springframework.cglib.proxy.MethodInterceptor; import org.springframework.cglib.proxy.MethodProxy; import com.mengma.dao.GoodsDao; import com.mengma.jdk.MyAspect; public class MyBeanFactory { public static GoodsDao getBean() { // 準備目標類 final GoodsDao goodsDao = new GoodsDao(); // 創建切面類實例 final MyAspect myAspect = new MyAspect(); // 生成代理類,CGLIB在運行時,生成指定對象的子類,增強 Enhancer enhancer = new Enhancer(); // 確定需要增強的類 enhancer.setSuperclass(goodsDao.getClass()); // 添加回調函數 enhancer.setCallback(new MethodInterceptor() { // intercept 相當于 jdk invoke,前三個參數與 jdk invoke—致 @Override public Object intercept(Object proxy, Method method, Object[] args, MethodProxy methodProxy) throws Throwable { myAspect.myBefore(); // 前增強 Object obj = method.invoke(goodsDao, args); // 目標方法執行 myAspect.myAfter(); // 后增強 return obj; } }); // 創建代理類 GoodsDao goodsDaoProxy = (GoodsDao) enhancer.create(); return goodsDaoProxy; } }
上述代碼中,應用了 CGLIB 的核心類 Enhancer。在第 19 行代碼調用了 Enhancer 類的 setSuperclass() 方法,確定目標對象。
第 21 行代碼調用 setCallback() 方法添加回調函數;第 24 行代碼的 intercept() 方法相當于 JDK 動態代理方式中的 invoke() 方法,該方法會在目標方法執行的前后,對切面類中的方法進行增強;第 33~34 行代碼調用 Enhancer 類的 create() 方法創建代理類,最后將代理類返回。
3. 創建測試類
在 com.mengma.cglib 包下創建測試類 CGLIBProxyTest,編輯后如下所示。
package com.mengma.cglib; import org.junit.Test; import com.mengma.dao.GoodsDao; public class CGLIBProxyTest { @Test public void test() { // 從工廠獲得指定的內容(相當于spring獲得,但此內容時代理對象) GoodsDao goodsDao = MyBeanFactory.getBean(); // 執行方法 goodsDao.add(); goodsDao.update(); goodsDao.delete(); goodsDao.find(); } }
上述代碼中,調用 getBean() 方法時,依然獲取的是 goodsDao 的代理對象,然后調用該對象的方法。使用 JUnit 測試運行 test() 方法,運行成功后,控制臺的輸出結果如圖 2 所示。
圖 2 輸出結果
從圖 2 的輸出結果中可以看出,在調用目標類的方法前后,也成功調用了增強的代碼,由此說明,使用 CGLIB 代理的方式同樣實現了手動代理。
以上就是本文的全部內容,希望對大家的學習有所幫助,也希望大家多多支持億速云。
免責聲明:本站發布的內容(圖片、視頻和文字)以原創、轉載和分享為主,文章觀點不代表本網站立場,如果涉及侵權請聯系站長郵箱:is@yisu.com進行舉報,并提供相關證據,一經查實,將立刻刪除涉嫌侵權內容。