当前位置: 首页 > news >正文

【JDK动态代理】及【CGLib动态代理】:Java的两种动态代理方式

Java的两种动态代理方式

  • 动态代理是什么?
  • JDK动态代理
  • CGLib动态代理
      • CGLib 底层原理
      • CGLib 实现步骤
  • 两者区别
  • Spring AOP原理--动态代理

动态代理是什么?

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

代理类在程序运行期间,创建的代理对象称之为动态代理对象。

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

创建代理对象的两个方法:

JDK动态代理

Proxy.newProxyInstance(三个参数);

CGLib动态代理

Enhancer.create(两个参数);

JDK动态代理

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

public class LogJdkProxy {
    /**
     * 生成对象的代理对象,对被代理对象进行所有方法日志增强
     * 参数:原始对象
     * 返回值:被代理的对象
     * 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
🔺使用CGLib的Enhancer类创建代理对象
🔺注意:如果报 asmxxxx 异常,需要导入 asm.jar包

public class LogCGLibProxy {
    /**
     * 生成对象的代理对象,对被代理对象进行所有方法日志增强
     * 参数:原始对象
     * 返回值:被代理的对象
     * 使用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 的区别主要体现在以下方面:

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

Spring AOP原理–动态代理

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

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

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

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

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

4.AOP 使用场景

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

相关文章:

  • MySQL数据同步到 Redis 缓存的几种方法
  • 一个nginx的小项目,不写代码,实现在局域网内访问其他电脑的网页
  • 基于Reactor模式下的epoll多路复用服务器
  • 为什么软件测试面试了几个月都没有offer,从HR角度分析
  • IO多路复用--[select | poll | epoll | Reactor]
  • java面试八股文之------Java并发夺命23问
  • 23.3.14打卡 2022年江西省大学生程序设计竞赛(正式赛)ABL
  • FPGA实现CSI-2 解码MIPI视频 2line 720P分辨率 OV5647采集 提供工程源码和技术支持
  • 改进YOLO系列 | CVPR2023最新Backbone | FasterNet 远超 ShuffleNet、MobileNet、MobileViT 等模型
  • 蓝桥杯刷题冲刺 | 倒计时28天
  • 改进 YOLO V5 的密集行人检测算法研究(论文研读)——目标检测
  • GPT4来了,多模态模型上线
  • 【C/C++】必知必会知识点大总结
  • css 画图之质感盒子
  • C/C++网络编程笔记Socket
  • [ JavaScript ] 数据结构与算法 —— 链表
  • [deviceone开发]-do_Webview的基本示例
  • [笔记] php常见简单功能及函数
  • C++类中的特殊成员函数
  • Cumulo 的 ClojureScript 模块已经成型
  • Fastjson的基本使用方法大全
  • jQuery(一)
  • MySQL几个简单SQL的优化
  • vue-loader 源码解析系列之 selector
  • 成为一名优秀的Developer的书单
  • 读懂package.json -- 依赖管理
  • 服务器从安装到部署全过程(二)
  • 力扣(LeetCode)357
  • 浅析微信支付:申请退款、退款回调接口、查询退款
  • 我的业余项目总结
  • ​LeetCode解法汇总2182. 构造限制重复的字符串
  • #etcd#安装时出错
  • (6)添加vue-cookie
  • (Demo分享)利用原生JavaScript-随机数-实现做一个烟花案例
  • (LNMP) How To Install Linux, nginx, MySQL, PHP
  • (Matalb时序预测)WOA-BP鲸鱼算法优化BP神经网络的多维时序回归预测
  • (二十三)Flask之高频面试点
  • (紀錄)[ASP.NET MVC][jQuery]-2 純手工打造屬於自己的 jQuery GridView (含完整程式碼下載)...
  • (十八)用JAVA编写MP3解码器——迷你播放器
  • (译)计算距离、方位和更多经纬度之间的点
  • .net core 源码_ASP.NET Core之Identity源码学习
  • .Net 知识杂记
  • .Net实现SCrypt Hash加密
  • .NET值类型变量“活”在哪?
  • @Conditional注解详解
  • [12] 使用 CUDA 加速排序算法
  • [20180129]bash显示path环境变量.txt
  • [C++][数据结构][算法]单链式结构的深拷贝
  • [CISCN2021 Quals]upload(PNG-IDAT块嵌入马)
  • [CSDN首发]鱿鱼游戏的具体玩法详细介绍
  • [HTML]HTML5实现可编辑表格
  • [I2C]I2C通信协议详解(二) --- I2C时序及规格指引
  • [kubernetes]控制平面ETCD
  • [leetcode] Balanced Binary Tree
  • [one_demo_4]不使用第3个变量交换两个变量的值