Spring CGLIB動態代理演練
CGLIB(Code Generation Library)是一個高性能開源的代碼生成包,它被許多 AOP 框架所使用,其底層是通過使用一個小而快的字節碼處理框架 ASM(Java 字節碼操控框架)轉換字節碼並生成新的類。而 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 {
// TODO Auto-generated method stub
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 的代理對象,然後調用該對象的方法。
4. 運行項目並查看結果
使用 JUnit 測試運行 CGLIBProxyTest,運行成功後,控制檯的輸出結果如圖:
輸出結果中可以看出,在調用目標類的方法前後,也成功調用了增強的代碼,由此說明,使用 CGLIB 代理的方式同樣實現了動態代理。