什么是动态代理?两种常用的动态代理方式_qq_16570607的博客-程序员秘密

技术标签: 动态代理  

什么是动态代理?

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

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

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

Proxy.newProxyInstance(三个参数);

//CGLib动态代理

Enhancer.create(两个参数);

两种常用的动态代理方式

  1. 基于接口的动态代理

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

    • 提供者:第三方 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 底层原理

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

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

CGLib 实现步骤

  1. 创建一个实现接口 MethodInterceptor 的代理类,重写 intercept 方法;
  2. 创建获取被代理类的方法 getInstance(Object target);
  3. 获取代理类,通过代理调用方法。

两者区别

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

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

聊聊 Spring AOP原理

  1. AOP 思想

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

  1. AOP 作用

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

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

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

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

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

版权声明:本文为博主原创文章,遵循 CC 4.0 BY-SA 版权协议,转载请附上原文出处链接和本声明。
本文链接:https://blog.csdn.net/qq_16570607/article/details/118360338

智能推荐

智能服务器虚拟化,详解四大服务器虚拟化架构_约会师老马的博客-程序员秘密

在本指南中,你将了解到服务器虚拟化、操作系统虚拟化、主机式虚拟化(hosted virtualization)和裸机虚拟化(bare-metal virtualization)的一些细节和它们的差异。还可以了解一下叫做混合虚拟化(Hybrid Virtualization)的新型虚拟技术,以及Microsoft的2008计划如何影响虚拟领域。VMware发布ESX已 经有七年了,虽然VMwar...

openCV (七) 直方图和傅里叶变换 学习回顾记录_cv2图像频谱图_深度菜鸡-达闻西的博客-程序员秘密

文章目录一、直方图(1) 图像均衡化二、傅里叶变换(1)低通滤波(2)高通滤波一、直方图语法:cv2.calcHist(images,channels,mask,histSize,ranges)- images:原图像格式为uint8或float32,当传入函数时需要用[]括起来使用;- channels: 同样用中括号括起来,告诉程序统计图像的直方图。如果是灰度图,就为[0],如果是彩色图像,传入参数为[0][1],[2]对应着BGR- mask:掩模图像。如果要统计整幅图想的直方图,为Non

2023跨境出海:奥地利电商市场现状及发展前景_wmdage的博客-程序员秘密

提起奥地利,很多人的第一印象就是维也纳的金色大厅、莫扎特,还有阿尔卑斯山的壮丽美景。其实奥地利的电商市场一样很美丽,是欧洲新增的一片蓝海市场。这也使得不少跨境卖家争先进行布局,那奥地利的市场潜力究竟怎么样呢?本文Nox聚星就和大家好好聊一聊。

Java学习笔记9---类静态成员变量的存储位置及JVM的内存划分_weixin_30586257的博客-程序员秘密

笔记8提到了类静态成员变量的访问方式,但静态成员变量存储在哪里呢?在网上查阅不少资料,发现好多内容都是过时的了,其中主流观点是静态成员变量存放在方法区。JDK8之前,静态成员变量确实存放在方法区;但JDK8之后就取消了“永久代”,取而代之的是“元空间”,永久代中的数据也进行了迁移,静态成员变量迁移到了堆中(方法区是JVM的规范,永久代是方法区的具体实现)。作者: 蝉蝉请尊重作者劳...

C++从txt读取数据赋值给变量_c++调用txt文件数据给函数赋值_silver_lining7的博客-程序员秘密

把txt的每行的三个数据赋值到Name,Tel,Email:#include #include #include #include using namespace std;class UserData{friend ifstream& operator>>(ifstream& is, UserData& user);friend ostream...

随便推点

wget使用proxy的配置_ckykdwr884269的博客-程序员秘密

在~/.wgetrc中设定代理http_proxy = http://ip_or_domainname:80/ftp_proxy = http://ip_or_domainname:80/use_proxy...

linux tar permission,ubuntu下操作目录,出现Permission denied的解决办法_西红柿气象台的博客-程序员秘密

今天从一个目录下拷贝一份文件到另外一个目录下时:[email protected]:~$cp jdk-7u79-linux-x64.tar /usr/lib/jvm出现如下提示:Permission denied一、表面上看是因为权限不足,因此可以通过对usr文件夹进行授权的方式解决:[email protected]:~$sudo chmod -R 777 usr其中-R 是指级联应用到目录里的所有...

基于AVR单片机的AT24C01-512eeprom读写程序_ba_wang_mao的博客-程序员秘密

针对AT24Cxx系列eeprom存储器,写的时候有越页功能,不用考虑页边界,I2C用软件模拟实现,完善中…#define SDA1() PORTC|=1< #define SDA0() PORTC&=~(1< #define SDAout() DDRC|=1< #define SDAin() DDRC&=~(1< #define RSDA() PINC&(1< #define SCL1() PORTC|=1< #define S

(PAT乙级)1043 输出PATest(C语言实现)_陈粑粑的小白鞋的博客-程序员秘密

‘总结:思路是正确的,先统计PATest个字母的个数,但在输出上自己处理得不是很好。其实只要任何一个字母的个数大于0就会继续输出,所以可以用这个条件作为循环的判断,还有个数应该是>0输出相应的字母,而不是≠0输出,因为循环还未结束,可能变成负数继续输出该字母。...

Ajax的同步调用场景——使用同步Ajax在 onunload 事件时通知服务器_ajax 同步场景_jlds123的博客-程序员秘密

<br />在一种场景下,server端维护了浏览器客户端的状态信息,当浏览器关闭时需要立刻通知server用户已经离开了,server端好清理状态。这种情况的典型案例就是:浏览器客户端的即时通信,即web IM。因为用户关闭了窗口就表明用户已经不在线了,所有对话都结束了,server端和对话的另一方需要立刻知道这个状态变化。而一般情况下,server端并不需要立刻通知的情况,那就无所谓了,只要过期超时即可。要想在用户关闭浏览器后,server端得到通知,一般有两种做法:方法一:

推荐文章

热门文章

相关标签