CGLIB原理

news2024/10/5 11:14:19

CGLIB(Code Generation Library)是一个强大的字节码生成库,用于在运行时生成代理类。CGLIB 实现了动态代理,与 Java 的 Proxy 不同,它不要求目标类实现接口,而是通过生成目标类的子类来实现代理。这使得 CGLIB 可以代理那些没有实现接口的类。接下来,我们从底层原理分析 CGLIB 的工作机制。

1. CGLIB 的基本原理

CGLIB 主要通过字节码增强技术,在运行时动态生成代理类,它的核心是基于 ASM 库操作字节码,从而生成一个继承自目标类的代理类。这意味着,CGLIB 通过生成目标类的子类并重写方法来实现代理,而不是像 java.lang.reflect.Proxy 那样基于接口进行代理。

CGLIB 代理的基本流程可以总结为以下几点:

  • 创建代理类的子类:CGLIB 通过继承目标类来生成代理类。因此,CGLIB 不能代理 final 类,因为 final 类无法被继承。
  • 重写方法:CGLIB 代理类会重写目标类的非 final 方法,在重写的方法中加入自定义的拦截逻辑,例如方法增强、日志记录等。
  • MethodInterceptor:CGLIB 的核心拦截机制是通过 MethodInterceptor 接口来实现的,所有被代理的方法调用都会被拦截,并进入 intercept() 方法处理。

2. CGLIB 的工作流程

CGLIB 生成动态代理类的过程大致可以分为以下几个步骤:

步骤 1: 代理类的生成

CGLIB 的代理类是通过继承目标类来生成的,因此代理类实际上是目标类的子类。在代理类中,CGLIB 会重写目标类的非 final 方法,以便在方法调用时加入自定义逻辑。

步骤 2: 拦截方法的调用

CGLIB 代理类会将所有方法的调用委托给 MethodInterceptor。这个拦截器是 CGLIB 的核心组件,当代理对象的方法被调用时,MethodInterceptor 会拦截这个调用,并决定是否继续调用原始方法,或者返回自定义结果。

public class MyMethodInterceptor implements MethodInterceptor {
    @Override
    public Object intercept(Object obj, Method method, Object[] args, MethodProxy proxy) throws Throwable {
        System.out.println("Before method: " + method.getName());
        Object result = proxy.invokeSuper(obj, args);
        System.out.println("After method: " + method.getName());
        return result;
    }
}

在这个例子中,intercept 方法通过 proxy.invokeSuper() 调用目标类的原始方法,并在调用前后加入自定义的逻辑。

步骤 3: 使用 Enhancer 创建代理对象

CGLIB 提供了 Enhancer 类用于生成代理对象。通过设置被代理的目标类和自定义的 MethodInterceptorEnhancer 会动态生成代理类并返回代理对象。

Enhancer enhancer = new Enhancer();
enhancer.setSuperclass(TargetClass.class);
enhancer.setCallback(new MyMethodInterceptor());
TargetClass proxy = (TargetClass) enhancer.create();

Enhancercreate() 方法会生成目标类的代理子类,并返回代理实例。

步骤 4: 代理方法的调用

当客户端调用代理对象的方法时,调用会被 MethodInterceptor 拦截。在拦截器中,代理类可以在方法调用前后插入自定义逻辑,或者完全改变方法的执行结果。

cglib生成代理类源码分析

虽然生成的类源码不易直接查看,但其逻辑结构类似于下面的伪代码:

public class RealService$$EnhancerByCGLIB extends RealService {
    private MethodInterceptor interceptor;

    public RealService$$EnhancerByCGLIB(MethodInterceptor interceptor) {
        this.interceptor = interceptor;
    }

    @Override
    public void doSomething() {
        try {
            Method method = RealService.class.getMethod("doSomething");
            interceptor.intercept(this, method, null, null); // 调用拦截器
        } catch (Exception e) {
            e.printStackTrace();
        }
    }
}

这个代理类的结构说明:

  1. 继承目标类 RealService:CGLIB 生成的代理类是 RealService 的子类,因此它继承了所有 RealService的方法。
  2. 拦截器 interceptor:代理类持有一个 MethodInterceptor 实例,在方法调用时会调用 interceptorintercept 方法。
  3. 方法重写:代理类会重写父类的方法,在调用方法时先通过拦截器进行额外的处理,然后调用原方法逻辑(通过 invokeSuper 调用父类方法)。

3. CGLIB 与 JDK 动态代理的区别

  • 代理方式:JDK 动态代理基于接口代理,而 CGLIB 基于子类继承代理。这意味着 CGLIB 可以代理没有实现接口的类,而 JDK 代理只能代理实现了接口的类。
  • 性能:CGLIB 生成的代理类在首次调用时需要进行字节码生成和类加载,这使得初次创建代理类的速度比 JDK 动态代理慢。但在后续的调用中,CGLIB 的执行效率较高,因为代理类已经生成并被加载到内存中。
  • 可代理范围:JDK 动态代理只能代理实现接口的类,而 CGLIB 可以代理所有非 final 类,但不能代理 final 方法。

4. CGLIB 的底层原理 - ASM 字节码操作

CGLIB 的核心是通过 ASM 库直接操作字节码。ASM 是一个 Java 字节码操作框架,允许开发者动态生成、修改 Java 类。CGLIB 使用 ASM 生成代理类的字节码,并动态加载到 JVM 中。

ASM 字节码增强的基本原理:
  1. ClassReader:用于读取目标类的字节码。
  2. ClassWriter:用于生成新的字节码,通常通过继承或修改现有的类字节码生成新的类。
  3. MethodVisitor:用于访问方法的字节码指令,从而可以修改方法的实现。

在 CGLIB 中,当我们使用 Enhancer 生成代理类时,CGLIB 会通过 ASM 读取目标类的字节码,并根据 MethodInterceptor 的配置动态生成代理类的字节码。然后,JVM 会将这些动态生成的字节码加载到内存中,从而完成代理类的创建。

5. CGLIB 的应用场景

CGLIB 在以下场景中非常有用:

  • 没有实现接口的类的代理:当目标类没有实现任何接口时,JDK 动态代理无法工作,这时可以使用 CGLIB 来为目标类创建代理。
  • 方法增强:在方法调用的前后加入逻辑,如日志记录、事务管理等。
  • AOP 实现:CGLIB 是 Spring AOP 的核心实现之一,当需要为没有接口的类实现 AOP 时,Spring 使用 CGLIB 来生成代理。

6. CGLIB 的限制

  • 不能代理 final 类或 final 方法:因为 CGLIB 是通过生成子类来实现代理的,所以它无法代理 final 修饰的类或方法。
  • 内存开销:由于 CGLIB 动态生成了代理类的字节码,因此会占用更多的内存,代理的类越多,内存开销越大。
  • 初次代理的性能问题:首次代理时,CGLIB 需要生成字节码并加载到 JVM 中,这个过程相对较慢。不过后续调用性能较好。

7. 总结

CGLIB 动态代理基于子类继承,通过字节码操作技术(如 ASM)动态生成代理类。它可以在运行时增强类的功能,使得开发者无需修改原有类的代码就能实现日志、权限校验等功能。虽然 CGLIB 在性能上有一定的开销,但其强大的代理能力使得它在 Spring 等框架中得到了广泛应用。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/2189770.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

【人工智能深度学习应用】妙策API最佳实践

功能概述 在文章创作过程中,用户可以借助AI妙策来辅助创作。AI妙策主要集中在聚合热点话题榜和平台话题榜两个方面。 具体功能 话题分析 AI妙策中的话题分析可以作为创作的灵感来源,通过网页视角选题、热门视角选题、时效视角选题、新颖视角选题&…

网站建设中,如何处理多语言版本?

在全球化的今天,网站多语言版本的处理成为了一项基本要求,尤其是对于面向国际用户的企业来说。以下是一些关于网站建设中如何处理多语言版本问题的建议: 使用URL国际化:通过在URL中添加语言代码(如example.com/en/表示…

【文心智能体】猫咪用品购物指南搭建教程

前言 随着科技的飞速发展和人们生活水平的提高,越来越多的家庭开始养宠物,其中猫咪因其可爱、独立和温顺的性格而备受青睐。然而,面对市场上琳琅满目的猫咪用品,如何为自家的“喵星人”挑选最合适的商品,成为了许多宠…

vue3学习:axios输入城市名称查询该城市天气

说来惭愧,接触前端也有很长一段时间了,最近才学习axios与后端的交互。今天学习了一个查询城市天气的案例,只需输入城市名称,点击“查询”按钮便可以进行查询。运行效果如下: 案例只实现了基本的查询功能,没…

51单片机的金属探测器【proteus仿真+程序+报告+原理图+演示视频】

1、主要功能 该系统由AT89C51/STC89C52单片机LCD1602显示模块金属检测传感器继电器LED、蜂鸣器等模块构成。适用于金属探测仪、检测金属、剔除金属等相似项目。 可实现功能: 1、LCD1602实时显示是否检测到金属 2、金属检测传感器(按键模拟)检测是否有…

利用 Python 爬虫采集 1688商品详情

1688是中国的一个大型B2B电子商务平台,主要用于批发和采购各种商品。对于需要从1688上获取商品详情数据、工程数据或店铺数据的用户来说,可以采用以下几种常见的方法: 官方API接口:如果1688提供了官方的API接口,那么可…

Java | Leetcode Java题解之第456题132模式

题目&#xff1a; 题解&#xff1a; class Solution {public boolean find132pattern(int[] nums) {int n nums.length;List<Integer> candidateI new ArrayList<Integer>();candidateI.add(nums[0]);List<Integer> candidateJ new ArrayList<Integer…

【华为HCIP实战课程四】OSPF邻居关系排错时间和区域问题,网络工程师

一、OSPF邻居关系排错 1、MA网络(默认的以太网、FR和ATM)要求掩码一致 2、相邻OSPF设备RID相同不能建立邻居-----上节已经详细演示说明 3、同一链路必须相同区域 4、Hello和死亡时间匹配 5、MTU检测 6、认证 7、Flag位的一致性 8、华为设备上网络类型不一致 二、同一…

红日靶场1学习笔记

一、准备工作 1、靶场搭建 靶场地址 靶场描述 靶场拓扑图 其他相关靶场搭建详情见靶场地址相关说明 2、靶场相关主机信息 后续打靶场的过程中&#xff0c;如果不是短时间内完成&#xff0c;可能ip会有变化 主机ip密码角色win7192.168.122.131hongrisec2019!边界服务器win…

华为OD机试 - 垃圾短信识别(Python/JS/C/C++ 2024 E卷 100分)

华为OD机试 2024E卷题库疯狂收录中&#xff0c;刷题点这里 专栏导读 本专栏收录于《华为OD机试真题&#xff08;Python/JS/C/C&#xff09;》。 刷的越多&#xff0c;抽中的概率越大&#xff0c;私信哪吒&#xff0c;备注华为OD&#xff0c;加入华为OD刷题交流群&#xff0c;…

8645 归并排序(非递归算法)

### 思路 非递归归并排序通过逐步合并相邻的子数组来实现排序。每次合并后输出当前排序结果。 ### 伪代码 1. 读取输入的待排序关键字个数n。 2. 读取n个待排序关键字并存储在数组中。 3. 对数组进行非递归归并排序&#xff1a; - 初始化子数组的大小curr_size为1。 - 逐…

手机使用技巧:8 个 Android 锁屏移除工具 [解锁 Android]

有时候&#xff0c;您会被锁定在自己的 Android 设备之外&#xff0c;而且似乎不可能重新进入。 一个例子就是你买了一部二手手机&#xff0c;后来发现无法使用。另一种情况是你忘记了屏幕锁定密码和用于验证密码的 Google 帐户凭据。这种情况很少见&#xff0c;但确实会发生&…

vite学习教程01、vite构建vue2

文章目录 前言一、vite初始化项目二、修改配置文件2.1、修改main.js文件2.2、修改App.vue文件2.3、修改helloworld.vue2.4、修改vite.conf.js2.5、修改vue版本--修改package.json文件 三、安装vue2和vite插件四、启动服务资料获取 前言 博主介绍&#xff1a;✌目前全网粉丝3W&…

网站建设中,静态网页和动态网页分别是什么,有什么区别

在现代网页设计和开发中&#xff0c;理解静态网页与动态网页的区别是至关重要的。 静态网页 定义&#xff1a; 静态网页是指内容固定不变&#xff0c;用户访问时服务器直接返回存储在服务器上的HTML文件。这些页面通常以.html、.htm等扩展名结尾&#xff0c;不包含特殊符号如“…

多模态大模型调研

Clip的泛化能力超强 压缩图片 GPT4 NLP与CV的区分是历史阶段性的产物&#xff0c;有了vit/transformer之后&#xff0c;两者的区分不再明显 Glip 应用于目标检测

Hadoop之WordCount测试

1、Hadoop简介&#xff1a; Hadoop是Apache旗下的一个用Java语言实现的开源软件框架&#xff0c;是一个开发和运行处理大规模数据的软件平台。 Hadoop的核心组件包括Hadoop分布式文件系统&#xff08;HDFS&#xff09;和MapReduce编程模型。HDFS是一个高度容错的系统&#xf…

CSS3练习--电商web

免责声明&#xff1a;本文仅做分享&#xff01; 目录 小练--小兔鲜儿 目录构建 SEO 三大标签 Favicon 图标 布局网页 版心 快捷导航&#xff08;shortcut&#xff09; 头部&#xff08;header&#xff09; logo 导航 搜索 购物车 底部&#xff08;footer&#xff0…

Vue工程化开发

Vue工程化开发 一、工程化开发和脚手架 1.开发Vue的两种方式 核心包传统开发模式&#xff1a;基于html / css / js 文件&#xff0c;直接引入核心包&#xff0c;开发 Vue。工程化开发模式&#xff1a;基于构建工具&#xff08;例如&#xff1a;webpack&#xff09;的环境中开…

【2024版本】Mac/Windows IDEA安装教程

IDEA 2024版本真的很强大&#xff0c;此外JDK发布了最新稳定版 JDK21 &#xff0c;只有新版本支持JDK 21、JDK22。原来数据库插件不支持redis等一些NoSql的数据库的连接&#xff0c;如果要使用需要自己单独装收费的插件。直接打开idea就很吃内存了&#xff0c;再打开其他一大堆…

计算机毕业设计 网上书店系统的设计与实现 Java实战项目 附源码+文档+视频讲解

博主介绍&#xff1a;✌从事软件开发10年之余&#xff0c;专注于Java技术领域、Python人工智能及数据挖掘、小程序项目开发和Android项目开发等。CSDN、掘金、华为云、InfoQ、阿里云等平台优质作者✌ &#x1f345;文末获取源码联系&#x1f345; &#x1f447;&#x1f3fb; 精…