Java的CGLIB动态代理深入解析

 更新时间:2023年11月27日 09:53:54   作者:一码评川  
这篇文章主要介绍了Java的CGLIB动态代理深入解析,CGLIB是强大的、高性能的代码生成库,被广泛应用于AOP框架,它底层使用ASM来操作字节码生成新的类,为对象引入间接级别,以控制对象的访问,需要的朋友可以参考下

一、介绍

CGLIB是强大的、高性能的代码生成库,被广泛应用于AOP框架,它底层使用ASM来操作字节码生成新的类,为对象引入间接级别,以控制对象的访问。

CGLIB相比于JDK动态代理更加强大,JDK动态代理只能对接口进行代理,而CGLIB既可以代理普通类,也能够代理接口。

优点: 通过FastClass机制调用方法,比JDK动态代理的反射机制效率高;被代理类无需实现接口

缺点: 运行期生成字节码,通过ASM写Class字节码,效率低;不能对final类及final方法进行代理

二、工作原理

g

CGLIB 通过动态生成一个需要被代理类的子类(即被代理类作为父类),该子类重写被代理类的所有不是 final 修饰的方法,并在子类中采用方法拦截的技术拦截父类所有的方法调用,进而织入横切逻辑。此外,因为 CGLIB 采用整型变量建立了方法索引,这比使用 JDK 动态代理更快(使用 Java 反射技术创建代理类的实例)。

2.1步骤说明

生成代理对象 创建要被代理的类或接口–MyFly

public class MyFly implements Fly {
    @Override
    public void doFly() {
        System.out.println("wo的");
    }
}

实现MethodInterceptor并实现intercept方法 --CglibProxy

class CglibProxy implements MethodInterceptor {
    /**
     * @param o:           代理对象
     * @param method:      被代理方法
     * @param params:      方法入参
     * @param methodProxy: CGLIB方法
     **/
    @Override
    public Object intercept(Object o, Method method, Object[] params, MethodProxy methodProxy) throws Throwable {
        System.out.println("【增强方法】代理对象正在执行的方法:" + method.getName());
        Object result = methodProxy.invokeSuper(o, params);
        return result;
    }
}

创建Enhancer(设置要被代理的类和调用方法时触发的拦截器)

 public static Object creatCglibProxyObj(Class<?> clazz) {
        Enhancer enhancer = new Enhancer();
        // 为加强器指定要代理的业务类(即为下面生成的代理类指定父类)
        enhancer.setSuperclass(clazz);
        // 设置回调:对于代理类上所有方法的调用,都会调用CallBack,而Callback则需要实现intercept()方法
        enhancer.setCallback(new CglibProxy());
        return enhancer.create();
    }

执行程序

public static void main(String[] args) {
        System.setProperty(DebuggingClassWriter.DEBUG_LOCATION_PROPERTY, "D:\\class");
        Fly fly = (Fly) CglibProxyFactory.creatCglibProxyObj(MyFly.class);
        fly.doFly();
    }

结果:

在这里插入图片描述

2.2 代理对象分析

通过设置属性来输出生成的代理对象

System.setProperty(DebuggingClassWriter.DEBUG_LOCATION_PROPERTY, "D:\\class");

在这里插入图片描述

通过IDEA打开class文件

2.2.1表头分析

在这里插入图片描述

结论: 生成的代理对象继承了我们需要被代理的对象同时实现了Factory这里是通过继承的方式来代理的,所以他即可以代理类也可以代理接口因为是继承了被代理类,所以在java中子类是无法重写父类final方法的,这也就解释了为什么CGLIB无法代理final修饰的方法了

2.2.2 stataic 静态代码块

在这里插入图片描述

结论: 这里面创建了一些后续要试用的变量引用

2.2.3 重写父类方法

在这里插入图片描述

结论: 重写的方法被final,防止后面被修改这里的var10000就是我们在创建代理时传入的CglibProxy这里也就解释了CGLIB是如何动态增强方法的基本逻辑(在每次调用方法时都会先去调用MethodInterceptor的实现类中的intercept方法,然后我们只需要在intercept方法中实现要加强的代码即可)

2.3 代理方法调用过程分析

调用代理方法

在这里插入图片描述

结论: 创建代理对象fly通过调用代理对象fly.doFly()方法

在这里插入图片描述

3. 调用MethodInterceptor的intercept方法,这里就是调用我们的CglibProxy

在这里插入图片描述

4. 调用methodProxy.invokeSuper(o, params);这里就是要调用被代理类的原始方法

在这里插入图片描述

通过init()来初始化生成代理类和被代理类的FastClass

在这里插入图片描述

helper:

在这里插入图片描述

生成的FastClass文件

在这里插入图片描述

生成的FastClass文件的invoke方法

在这里插入图片描述

init完成后这里会继续调用fci.f2.invoke(fci.i2, obj, args); 这里的fci.f2就是刚刚生成的代理对象FastCalss对象

在这里插入图片描述

这里程序会传入17,调用代理类的CGLIB$doFly$0()方法;(大家可以debug看,每次生成的文件位置会不一样)

在这里插入图片描述

调用代理类的CGLIB$doFly$0() 到这里基本可以知道它是如何帮助我们调用被代理类的方法了这里的super指的就是MyFly,因为CGLIB生成的代理类继承了我们要被代理的类

在这里插入图片描述

到此这篇关于Java的CGLIB动态代理深入解析的文章就介绍到这了,更多相关CGLIB动态代理内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家!

相关文章

  • SpringBoot集成Swagger构建api文档的操作

    SpringBoot集成Swagger构建api文档的操作

    这篇文章主要介绍了SpringBoot集成Swagger构建api文档的操作,具有很好的参考价值,希望对大家有所帮助。一起跟随小编过来看看吧
    2020-12-12
  • MyBatis Generator生成代码及使用方式详解

    MyBatis Generator生成代码及使用方式详解

    这篇文章主要介绍了MyBatis Generator生成代码及使用方式的相关资料,非常不错,具有参考借鉴价值,需要的朋友可以参考下
    2016-09-09
  • Java中DecimalFormat用法及符号含义

    Java中DecimalFormat用法及符号含义

    DecimalFormat是NumberFormat的一个具体子类,用于格式化十进制数字。这篇文章介绍了DecimalFormat的用法及符号含义,需要的朋友可以收藏下,方便下次浏览观看
    2021-12-12
  • 关于mybatis-plus逻辑删除自动填充更新时间的问题

    关于mybatis-plus逻辑删除自动填充更新时间的问题

    mybatis-plus是对mybatis的增强,mybatis-plus更像是面向对象编程,数据库基本CRUD的操作可以不用手动编写SQL语句,大大提高了开发的效率,这篇文章主要介绍了mybatis-plus逻辑删除自动填充更新时间问题,需要的朋友可以参考下
    2022-07-07
  • Java Spring Security认证与授权及注销和权限控制篇综合解析

    Java Spring Security认证与授权及注销和权限控制篇综合解析

    Spring Security 是 Spring 家族中的一个安全管理框架,实际上,在 Spring Boot 出现之前,Spring Security 就已经发展了多年了,但是使用的并不多,安全管理这个领域,一直是 Shiro 的天下
    2021-10-10
  • scala 隐式转换与隐式参数的使用方法

    scala 隐式转换与隐式参数的使用方法

    这篇文章主要介绍了scala 隐式转换与隐式参数的使用方法,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2019-11-11
  • java基础--JDK SPI概述

    java基础--JDK SPI概述

    SPI是一种服务发现机制,本文就SPI做了详细概述,具有很好的参考价值,希望对小伙伴们有所帮助,感兴趣的朋友一起来参考参考吧
    2021-08-08
  • java的jdbc简单封装方法

    java的jdbc简单封装方法

    本篇文章是对java的jdbc简单封装方法进行了详细的分析介绍,需要的朋友参考下
    2015-07-07
  • 浅谈java中定义泛型类和定义泛型方法的写法

    浅谈java中定义泛型类和定义泛型方法的写法

    下面小编就为大家带来一篇浅谈java中定义泛型类和定义泛型方法的写法。小编觉得挺不错的,现在就分享给大家,也给大家做个参考。一起跟随小编过来看看吧
    2017-02-02
  • 详解 Java Maximum redirects (100) exceeded

    详解 Java Maximum redirects (100) exceeded

    这篇文章主要介绍了详解 Java Maximum redirects (100) exceeded的相关资料,需要的朋友可以参考下
    2017-05-05

最新评论