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 代理的方式同样实现了手动代理。
以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持小牛知识库。
本文向大家介绍Spring JDK动态代理实现过程详解,包括了Spring JDK动态代理实现过程详解的使用技巧和注意事项,需要的朋友参考一下 这篇文章主要介绍了Spring JDK动态代理实现过程详解,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友可以参考下 1. 创建项目 在 MyEclipse 中创建一个名称为 springDemo03 的 Web 项
本文向大家介绍Java简单实现动态代理模式过程解析,包括了Java简单实现动态代理模式过程解析的使用技巧和注意事项,需要的朋友参考一下 基本知识:JDK动态代理是java.lang.reflect.*包提供的方式,它必须借助一个接口才能生成代理对象. 步骤: 1、首先建一个接口类,并提供一个实现类: 2、创建一个代理类,必须要实现InvocationHandler类 3、测试类 结果: add方法
本文向大家介绍Python动态类型实现原理及过程解析,包括了Python动态类型实现原理及过程解析的使用技巧和注意事项,需要的朋友参考一下 在python中,我们使用变量时,并没有声明变量的存在和类型。类型是在运行过程中自动决定的。 a = 3 python将会执行三步去完成上面这个请求。 1.创建一个对象代表3 2.创建一个变量a,如果a未创建。 3.将变量a与对象3相连接。 可以将变量a看作对
本文向大家介绍MyBatis动态SQL实现配置过程解析,包括了MyBatis动态SQL实现配置过程解析的使用技巧和注意事项,需要的朋友参考一下 动态SQL 什么是动态SQL: 动态SQL就是根据不同的条件生成不同的SQL语句 if choose(when,otherwise) trim(where,set) foreach 1、搭建环境 建表 创建一个基础工程 导包 编写配置文件 编写实体类
本文向大家介绍怎么实现动态代理?相关面试题,主要包含被问及怎么实现动态代理?时的应答技巧和注意事项,需要的朋友参考一下 JDK 原生动态代理和 cglib 动态代理。JDK 原生动态代理是基于接口实现的,而 cglib 是基于继承当前类的子类实现的。
本文向大家介绍详解Java动态代理的实现机制,包括了详解Java动态代理的实现机制的使用技巧和注意事项,需要的朋友参考一下 一、概述 代理是一种设计模式,其目的是为其他对象提供一个代理以控制对某个对象的访问,代理类负责为委托类预处理消息,过滤消息并转发消息以及进行消息被委托类执行后的后续处理。为了保持行为的一致性,代理类和委托类通常会实现相同的接口。 按照代理的创建时期,代理类可分为两种: