在图 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—致@Overridepublic 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 {@Testpublic void test() {// 从工厂获得指定的内容(相当于spring获得,但此内容时代理对象)GoodsDao goodsDao = MyBeanFactory.getBean();// 执行方法goodsDao.add();goodsDao.update();goodsDao.delete();goodsDao.find();}}