什么是动态代理?两种常用的动态代理方式

您所在的位置:网站首页 模块包括什么和什么两种类型 什么是动态代理?两种常用的动态代理方式

什么是动态代理?两种常用的动态代理方式

2024-07-16 22:40| 来源: 网络整理| 查看: 265

什么是动态代理?

动态代理就是,在程序运行期,创建目标对象的代理对象,并对目标对象中的方法进行功能性增强的一种技术。在生成代理对象的过程中,目标对象不变,代理对象中的方法是目标对象方法的增强方法。可以理解为运行期间,对象中方法的动态拦截,在拦截方法的前后执行功能操作。

代理类在程序运行期间,创建的代理对象称之为动态代理对象。这种情况下,创建的代理对象,并不是事先在Java代码中定义好的。而是在运行期间,根据我们在动态代理对象中的“指示”,动态生成的。也就是说,你想获取哪个对象的代理,动态代理就会为你动态的生成这个对象的代理对象。动态代理可以对被代理对象的方法进行功能增强。有了动态代理的技术,那么就可以在不修改方法源码的情况下,增强被代理对象的方法的功能,在方法执行前后做任何你想做的事情。

创建代理对象的两个方法: //JDK动态代理

Proxy.newProxyInstance(三个参数);

//CGLib动态代理

Enhancer.create(两个参数);

两种常用的动态代理方式

基于接口的动态代理

提供者:JDK使用JDK官方的Proxy类创建代理对象注意:代理的目标对象必须实现接口

基于类的动态代理

提供者:第三方 CGLib使用CGLib的Enhancer类创建代理对象注意:如果报 asmxxxx 异常,需要导入 asm.jar包

以下是两种代理方式的实例代码:

public class LogProxy { /** * 生成对象的代理对象,对被代理对象进行所有方法日志增强 * 参数:原始对象 * 返回值:被代理的对象 * JDK 动态代理 * 基于接口的动态代理 * 被代理类必须实现接口 * JDK提供的 */ public static Object getObject(final Object obj){ /** * 创建对象的代理对象 * 参数一:类加载器 * 参数二:对象的接口 * 参数三:调用处理器,代理对象中的方法被调用,都会在执行方法。对所有被代理对象的方法进行拦截 */ Object proxyInstance = Proxy.newProxyInstance(obj.getClass().getClassLoader() , obj.getClass().getInterfaces(), new InvocationHandler() { public Object invoke(Object proxy, Method method, Object[] args) throws Throwable { //方法执行前 long startTime = System.currentTimeMillis(); Object result = method.invoke(obj, args);//执行方法的调用 //方法执行后 long endTime = System.currentTimeMillis(); SimpleDateFormat sdf = new SimpleDateFormat(); System.out.printf(String.format("%s方法执行结束时间:%%s ;方法执行耗时:%%d%%n" , method.getName()), sdf.format(endTime), endTime - startTime); return result; } }); return proxyInstance; } /** * 使用CGLib创建动态代理对象 * 第三方提供的的创建代理对象的方式CGLib * 被代理对象不能用final修饰 * 使用的是Enhancer类创建代理对象 */ public static Object getObjectByCGLib(final Object obj){ /** * 使用CGLib的Enhancer创建代理对象 * 参数一:对象的字节码文件 * 参数二:方法的拦截器 */ Object proxyObj = Enhancer.create(obj.getClass(), new MethodInterceptor() { public Object intercept(Object o, Method method, Object[] objects, MethodProxy methodProxy) throws Throwable { //方法执行前 long startTime = System.currentTimeMillis(); Object invokeObject = method.invoke(obj, objects);//执行方法的调用 //方法执行后 long endTime = System.currentTimeMillis(); SimpleDateFormat sdf = new SimpleDateFormat(); System.out.printf(String.format("%s方法执行结束时间:%%s ;方法执行耗时:%%d%%n" , method.getName()), sdf.format(endTime), endTime - startTime); return invokeObject; } }); return proxyObj; } } CGLib 底层原理

通过查看 Enhancer 类源码,最终也是生成动态代理类的字节码,动态代理类继承要被代理的类,然后实现其方法。

和 JDK Proxy 的实现代码比较类似,都是通过实现代理器的接口,再调用某一个方法完成动态代理的,唯一不同的是,CGLib 在初始化被代理类时,是通过 Enhancer 对象把代理对象设置为被代理类的子类来实现动态代理的。

CGLib 实现步骤 创建一个实现接口 MethodInterceptor 的代理类,重写 intercept 方法;创建获取被代理类的方法 getInstance(Object target);获取代理类,通过代理调用方法。 两者区别

JDK Proxy 和 CGLib 的区别主要体现在以下方面:

JDK Proxy 是 Java 语言自带的功能,无需通过加载第三方类实现;Java 对 JDK Proxy 提供了稳定的支持,并且会持续的升级和更新,Java 8 版本中的 JDK Proxy 性能相比于之前版本提升了很多;JDK Proxy 是通过拦截器加反射的方式实现的;JDK Proxy 只能代理实现接口的类;JDK Proxy 实现和调用起来比较简单;CGLib 是第三方提供的工具,基于 ASM 实现的,性能比较高;CGLib 无需通过接口来实现,它是针对类实现代理,主要是对指定的类生成一个子类,它是通过实现子类的方式来完成调用的。 聊聊 Spring AOP原理 AOP 思想

基于动态代理思想,对原来目标对象创建代理对象,在不修改原对象代码情况下,通过代理对象调用增强功能的代码,从而对原有业务方法进行增强。

AOP 作用

在不修改源代码的情况下,可以增加额外的功能,实现在原有功能基础上的增强。

AOP 使用场景 记录日志(调用方法后记录日志)监控性能(统计方法运行时间)权限控制(调用方法前校验是否有权限)事务管理(调用方法前开启事务,调用方法后提交关闭事务 )缓存优化(第一次调用查询数据库,将查询结果放入内存对象, 第二次调用,直接从内存对象返回,不需要查询数据库 ) AOP 实现原理

Spring AOP 的有两种实现方式:JDK proxy 和 CGLib 动态代理

当 Bean 实现接口时,Spring 使用 JDK proxy实现。当 Bean 没有实现接口时,Spring 使用 CGlib 代理实现。

通过配置可以强制使用 CGlib 代理(在 spring 配置中加入 aop:aspectj-autoproxy proxy-target-class=“true”)。



【本文地址】


今日新闻


推荐新闻


CopyRight 2018-2019 办公设备维修网 版权所有 豫ICP备15022753号-3