Spring CGLlB動態(tài)代理實現(xiàn)過程解析
JDK 動態(tài)代理使用起來非常簡單,但是它也有一定的局限性,這是因為 JDK 動態(tài)代理必須要實現(xiàn)一個或多個接口,如果不希望實現(xiàn)接口,則可以使用 CGLIB 代理。
CGLIB(Code Generation Library)是一個高性能開源的代碼生成包,它被許多 AOP 框架所使用,其底層是通過使用一個小而快的字節(jié)碼處理框架 ASM(Java 字節(jié)碼操控框架)轉(zhuǎn)換字節(jié)碼并生成新的類。因此 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 版本的核心包已經(jīng)集成了 CGLIB 所需要的包,所以在開發(fā)中不需要另外導(dǎo)入 ASM 的 JAR 包了。下面通過案例演示實現(xiàn) CGLIB 的代理過程。
1. 創(chuàng)建目標(biāo)類 GoodsDao
在 com.mengma.dao 包下創(chuàng)建目標(biāo)類 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. 創(chuàng)建代理類 MyBeanFactory
在 src 目錄下創(chuàng)建一個名為 com.mengma.cglib 的包,該包下創(chuàng)建類 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() {
// 準(zhǔn)備目標(biāo)類
final GoodsDao goodsDao = new GoodsDao();
// 創(chuàng)建切面類實例
final MyAspect myAspect = new MyAspect();
// 生成代理類,CGLIB在運行時,生成指定對象的子類,增強
Enhancer enhancer = new Enhancer();
// 確定需要增強的類
enhancer.setSuperclass(goodsDao.getClass());
// 添加回調(diào)函數(shù)
enhancer.setCallback(new MethodInterceptor() {
// intercept 相當(dāng)于 jdk invoke,前三個參數(shù)與 jdk invoke—致
@Override
public Object intercept(Object proxy, Method method, Object[] args,
MethodProxy methodProxy) throws Throwable {
myAspect.myBefore(); // 前增強
Object obj = method.invoke(goodsDao, args); // 目標(biāo)方法執(zhí)行
myAspect.myAfter(); // 后增強
return obj;
}
});
// 創(chuàng)建代理類
GoodsDao goodsDaoProxy = (GoodsDao) enhancer.create();
return goodsDaoProxy;
}
}
上述代碼中,應(yīng)用了 CGLIB 的核心類 Enhancer。在第 19 行代碼調(diào)用了 Enhancer 類的 setSuperclass() 方法,確定目標(biāo)對象。
第 21 行代碼調(diào)用 setCallback() 方法添加回調(diào)函數(shù);第 24 行代碼的 intercept() 方法相當(dāng)于 JDK 動態(tài)代理方式中的 invoke() 方法,該方法會在目標(biāo)方法執(zhí)行的前后,對切面類中的方法進(jìn)行增強;第 33~34 行代碼調(diào)用 Enhancer 類的 create() 方法創(chuàng)建代理類,最后將代理類返回。
3. 創(chuàng)建測試類
在 com.mengma.cglib 包下創(chuàng)建測試類 CGLIBProxyTest,編輯后如下所示。
package com.mengma.cglib;
import org.junit.Test;
import com.mengma.dao.GoodsDao;
public class CGLIBProxyTest {
@Test
public void test() {
// 從工廠獲得指定的內(nèi)容(相當(dāng)于spring獲得,但此內(nèi)容時代理對象)
GoodsDao goodsDao = MyBeanFactory.getBean();
// 執(zhí)行方法
goodsDao.add();
goodsDao.update();
goodsDao.delete();
goodsDao.find();
}
}
上述代碼中,調(diào)用 getBean() 方法時,依然獲取的是 goodsDao 的代理對象,然后調(diào)用該對象的方法。使用 JUnit 測試運行 test() 方法,運行成功后,控制臺的輸出結(jié)果如圖 2 所示。

圖 2 輸出結(jié)果
從圖 2 的輸出結(jié)果中可以看出,在調(diào)用目標(biāo)類的方法前后,也成功調(diào)用了增強的代碼,由此說明,使用 CGLIB 代理的方式同樣實現(xiàn)了手動代理。
以上就是本文的全部內(nèi)容,希望對大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。
相關(guān)文章
java實現(xiàn)把對象數(shù)組通過excel方式導(dǎo)出的功能
本文主要介紹了java實現(xiàn)把對象數(shù)組通過excel方式導(dǎo)出的功能的相關(guān)知識。具有很好的參考價值,下面跟著小編一起來看下吧2017-03-03
圖解Springboot集成七牛云并實現(xiàn)圖片上傳功能過程
在實際開發(fā)中 ,基本都會有應(yīng)用到文件上傳的場景,但隨著或多或少的需求問題,之前有在springboot上用過七牛云實現(xiàn)圖片上傳,今天因為某些原因又重新使用了下七牛云因此想總結(jié)下七牛云2021-11-11
使用@DS輕松解決動態(tài)數(shù)據(jù)源的問題
這篇文章主要介紹了使用@DS輕松解決動態(tài)數(shù)據(jù)源的問題,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教2022-05-05
java Socket無法完全接收返回內(nèi)容的解決方案
這篇文章主要介紹了java Socket無法完全接收返回內(nèi)容的解決方案,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教2021-10-10
Springboot使用Logback實現(xiàn)日志配置與異常記錄
默認(rèn)情況下,SpringBoot內(nèi)部使用logback作為系統(tǒng)日志實現(xiàn)的框架,將日志輸出到控制臺,不會寫到日志文件。本篇文章主要講解下如何自定義logabck.xml以及對logback文件中配置做一個詳解,需要的可以參考一下2022-11-11
Spring Security OAuth2實現(xiàn)使用JWT的示例代碼
這篇文章主要介紹了Spring Security OAuth2實現(xiàn)使用JWT的示例代碼,小編覺得挺不錯的,現(xiàn)在分享給大家,也給大家做個參考。一起跟隨小編過來看看吧2018-09-09

