cglib实现动态代理_cglib和jdk动态代理

cglib实现动态代理_cglib和jdk动态代理一、前言  说到动态代理,开发者们第一时间想到的就是JDK动态代理和cglib动态代理。了解Spring的同学应该知道,SpringAOP功能的底层实现,就是使用的这两种动态代理。两者区别JDK的动态代理机制只能代理实现了接口的类,而没有实现接口的类就不能实现JDK的动态代理;cglib动态代理是针对类来实现代理的,它的原理是对指定的目标类生成一个子类,并覆盖其中方法实现增强。使用cglib实现动态代理,完全不受代理类必须实现接口的限制。cglib底层采用ASM字节码生成框架,使用字节码技术生

大家好,又见面了,我是你们的朋友全栈君。如果您正在找激活码,请点击查看最新教程,关注关注公众号 “全栈程序员社区” 获取激活教程,可能之前旧版本教程已经失效.最新Idea2022.1教程亲测有效,一键激活。

Jetbrains全系列IDE稳定放心使用

一、前言

  说到动态代理,开发者们第一时间想到的就是JDK动态代理cglib动态代理。了解Spring的同学应该知道,Spring AOP功能的底层实现,就是使用的这两种动态代理。

两者区别
  • JDK的动态代理机制只能代理实现了接口的类,而没有实现接口的类就不能实现JDK的动态代理;
  • cglib动态代理是针对类来实现代理的,它的原理是对指定的目标类生成一个子类,并覆盖其中方法实现增强。使用cglib实现动态代理,完全不受代理类必须实现接口的限制。
  • cglib底层采用ASM字节码生成框架,使用字节码技术生成代理类,比使用Java反射效率要高。

因为cglib动态代理采用的是继承,所以不能对final修饰的类进行代理

二、使用场景

  “代理”二字,从字面意思上来看,就是代替目标类做一些预处理的事情。常用来把一些通用的与业务逻辑无关的逻辑放到代理类中处理。如:事务管理、参数校验、统计接口访问量、调用前后打印日志等等。

三、主要组件

  • MethodInterceptor:方法拦截器
    在调用目标方法时,cglib会回调MethodInterceptor接口方法,来实现你自己的代理逻辑,类似于JDK中的InvocationHandler接口。
/** * proxy:代理对象,CGLib动态生成的代理类实例 * method:目标对象的方法,上文中实体类所调用的被代理的方法引用 * args:目标对象方法的参数列表,参数值列表 * methodProxy:代理对象的方法,生成的代理类对方法的代理引用 */
 public Object intercept(Object obj, java.lang.reflect.Method method, Object[] args,
                           MethodProxy proxy) throws Throwable;

可以使用methodProxy.invokeSuper(obj,arg)来调用代理类实例上的proxy方法的父类方法,这样比反射调用方法快!

  • Enhancer:字节码增强器
    用来关联目标类代理处理逻辑类,并创建代理实例。

四、示例

需求:在进入方法前开启事务,方法调用结束后关闭事务。
由于cglib是一个第三方的框架,不是JDK自带的,所以要引入maven依赖。

<dependency>
    <groupId>cglib</groupId>
    <artifactId>cglib</artifactId>
    <version>3.2.6</version>
</dependency>
  1. 定义目标类:UserService
public class UserService { 
   
    public String getUserName(Long userId) { 
   
        System.out.println("获取用户名..");
        return "user" + userId;
    }
}
  1. 实现MethodInterceptor,定义事务拦截器
public class TransactionInterceptor implements MethodInterceptor { 
   
    Object target;

    public TransactionInterceptor(Object target) { 
   
        this.target = target;
    }

    /** * proxy:代理对象,CGLib动态生成的代理类实例 * method:目标对象的方法,上文中实体类所调用的被代理的方法引用 * args:目标对象方法的参数列表,参数值列表 * methodProxy:代理对象的方法,生成的代理类对方法的代理引用 */
    @Override
    public Object intercept(Object proxy, Method method, Object[] args, MethodProxy methodProxy) throws Throwable { 
   
        System.out.println("开启事务..." + proxy.getClass().getSimpleName());
        Object objValue = null;
        try { 
   
            // 反射调用目标类方法
            objValue = method.invoke(target, args);
            System.out.println("返回值为:" + objValue);
        } catch (Exception e) { 
   
            System.out.println("调用异常!" + e.getMessage());
        } finally { 
   
            System.out.println("调用结束,关闭事务...");
        }
        return objValue;
    }

    /** * 获取代理实例 */
    public Object getTargetProxy() { 
   
        // Enhancer类是cglib中的一个字节码增强器,它可以方便的为你所要处理的类进行扩展
        Enhancer eh = new Enhancer();
        // 1.将目标对象所在的类作为Enhancer类的父类
        eh.setSuperclass(target.getClass());
        // 2.通过实现MethodInterceptor实现方法回调
        eh.setCallback(this);
        // 3. 创建代理实例
        return eh.create();
    }
}

一般把获取代理实例的方法,也放在自定义的MethodInterceptor中。

  1. 使用
public static void main(String[] args) { 
   
    // 1. 创建目标实例
    UserService userService = new UserService();
    // 2. 创建事务拦截器
    TransactionInterceptor transactionInterceptor = new TransactionInterceptor(userService);
    // 3. 创建代理实例
    UserService userServiceProxy = (UserService) transactionInterceptor.getTargetProxy();
    // 4. 使用代理实例调用目标方法
    userServiceProxy.getUserName(6L);
}

在这里插入图片描述
这样便达到了代理的效果,开启、关闭事务的代码模块化到自定义的MethodInterceptor中,与UserService中的代码完全解耦!快去试试吧!

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请联系我们举报,一经查实,本站将立刻删除。

发布者:全栈程序员-站长,转载请注明出处:https://javaforall.net/180061.html原文链接:https://javaforall.net

(0)
全栈程序员-站长的头像全栈程序员-站长


相关推荐

  • Opacity 属性引发的层叠问题

    Opacity 属性引发的层叠问题Opacity属性引发的层叠问题总结

    2022年5月26日
    49
  • 什么是大数据架构?需要学什么内容?[通俗易懂]

    什么是大数据架构?需要学什么内容?[通俗易懂]大数据架构设计用来处理对传统数据库系统而言太大或太复杂的数据的引入、处理和分析。组织进入大数据领域的门槛各不相同,具体取决于用户的权限及其工具的功能。对某些组织来说,大数据可能意味着数百个GB的数据,而对另一些组织来说,大数据则意味着数百个TB的数据。随着处理大数据集的工具的发展,大数据的涵义也在不断地变化。慢慢地,这个术语更多的是指通过高级分析从数据集获取的价值,而不是严格地指数据的大小…

    2022年5月16日
    36
  • JS新规范padStart()详解,自己实现一个简单的padStart()

    JS新规范padStart()详解,自己实现一个简单的padStart()ES2017引入了字符串补全长度的功能。如果某个字符串不够指定长度,会在头部或尾部补全。padStart()用于头部补全,padEnd()用于尾部补全。

    2025年10月15日
    2
  • Esp8266学习之旅① 搭建开发环境,开始一个“hellow world”串口打印。

    Esp8266学习之旅① 搭建开发环境,开始一个“hellow world”串口打印。本系列博客学习由非官方人员半颗心脏潜心所力所写,仅仅做个人技术交流分享,不做任何商业用途。如有不对之处,请留言,本人及时更改。1、Esp8266之搭建开发环境,开始一个“hellowworld”串口打印。2、Esp8266之利用GPIO开始使用按钮点亮你的“第一盏灯”。3、Esp8266之利用“软件定时器”定时0.5秒闪烁点亮一盏LED。4、Esp8266之了解P

    2022年5月30日
    55
  • 红色故障码大全_图论的最短路问题

    红色故障码大全_图论的最短路问题原题链接战争中保持各个城市间的连通性非常重要。本题要求你编写一个报警程序,当失去一个城市导致国家被分裂为多个无法连通的区域时,就发出红色警报。注意:若该国本来就不完全连通,是分裂的k个区域,而失去一个城市并不改变其他城市之间的连通性,则不要发出警报。输入格式:输入在第一行给出两个整数N(0 < N ≤ 500)和M(≤ 5000),分别为城市个数(于是默认城市从0到N-1编号)和连接两城市的通路条数。随后M行,每行给出一条通路所连接的两个城市的编号,其间以1个空格分隔。在城市信息之后给出被攻占的

    2022年8月8日
    1
  • 数学建模之方差分析模型_数学建模层次分析法

    数学建模之方差分析模型_数学建模层次分析法《数学建模算法与应用》学习小结(11-20章)方差分析(AnalysisofVariance,ANOVA)应用场景单因素方差分析双因素方差分析方差分析(AnalysisofVariance,ANOVA)——用数理统计分析试验结果,鉴别各因素对结果影响程度的方法。人们关心的试验结果称为指标,试验中需要考察、可以控制的条件称为因素或因子,因素所处的状态称为水平应用场景为了使生产过…

    2022年10月15日
    3

发表回复

您的邮箱地址不会被公开。 必填项已用 * 标注

关注全栈程序员社区公众号