CGLIB动态代理(AOP原理)(面试重点)

推荐先看JDK 动态代理(Spring AOP 的原理)(面试重点)

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

JDK 动态代理有⼀个最致命的问题是其只能代理实现了接⼝的类.

有些场景下,我们的业务代码是直接实现的,并没有接⼝定义.为了解决这个问题,我们可以⽤CGLIB 动态代理机制来解决.

CGLIB(Code Generation Library)是⼀个基于ASM的字节码⽣成库,它允许我们在运⾏时对字节码进⾏ 修改和动态⽣成. CGLIB 通过继承⽅式实现代理, 很多知名的开源框架都使⽤到了CGLIB.例如 Spring 中的 AOP 模块中: 如果⽬标对象实现了接⼝,则默认采⽤ JDK 动态代理, 否则采⽤ CGLIB 动态代理.

CGLIB 动态代理类实现步骤

1.添加依赖

和 JDK 动态代理不同, CGLIB(Code Generation Library) 实际是属于⼀个开源项⽬,如果你要使⽤它的话,需要⼿动添加相关依赖

java 复制代码
<dependency>
 <groupId>cglib</groupId>
 <artifactId>cglib</artifactId>
 <version>3.3.0</version>
</dependency>

2. 定义⼀个类(被代理类 -> 目标类)

目标类实现的接口 :

java 复制代码
//声明房东(目标对象)要执行的相关操作
public interface HouseSubject {
    void rentHouse();   //出租房子
    void saleHouse();   //卖房子
}

目标类:

java 复制代码
public class Landlord implements HouseSubject{
    @Override
    public void rentHouse() {
        System.out.println("房东出租房子");
    }

    @Override
    public void saleHouse() {
        System.out.println("房东卖房子");
    }
}

3. ⾃定义 MethodInterceptor 并重写 intercept ⽅法, intercept ⽤于增强⽬标⽅法,和 JDK 动态代理中的 invoke ⽅法类似(写代理对象的逻辑)

java 复制代码
// CGLIB 动态代理的逻辑
public class CGLIBDynamicProxy implements MethodInterceptor {
    private Object target;

    public CGLIBDynamicProxy(Object target){
        this.target=target;
    }
    @Override
    public Object intercept(Object o, Method method, Object[] objects, MethodProxy methodProxy) throws Throwable {
        //动态代理的逻辑
        System.out.println("代理接手,工作开始");
        //执行目标对象中的操作
        Object result=methodProxy.invoke(target,objects);
        System.out.println("代理离手,工作结束");
        return result;
    }
}

4. 通过 Enhancer 类的 create() 创建并使用代理类

java 复制代码
//创建代理对象并使用
public class DynamicMain {
    public static void main(String[] args) {

        HouseSubject target=new Landlord();
        Landlord target1=new Landlord();

        //通过 CGLIB 创建代理对象
        HouseSubject proxy1= (HouseSubject) Enhancer.create(
                target.getClass(),
                new CGLIBDynamicProxy(target));

        Landlord proxy2=(Landlord) Enhancer.create(
                target1.getClass(),
                new CGLIBDynamicProxy(target1));

        //使用代理对象
        proxy1.saleHouse();
        proxy1.rentHouse();

        proxy2.saleHouse();
        proxy2.rentHouse();
    }
}

得到的结果如下:

成功进行了代理操作

相关推荐
狼爷9 小时前
吃透 Java Function 接口,搞定 99% 的 Stream 场景
java·函数式编程
葫芦和十三9 小时前
图解 MongoDB 15|journal 与持久化:写入怎么不丢,崩溃怎么恢复
后端·mongodb·面试
葫芦和十三9 小时前
图解 MongoDB 16|压缩:snappy、zstd 和 zlib 的取舍
后端·mongodb·面试
祎雪双十Gy13 小时前
从 DataX 的配置加载说起:我用 FastJson2 做了一个轻量级动态配置管理库
java·后端
labixiong13 小时前
实现一个能跑的迷你版Promise(一)
前端·javascript·面试
小锋java123413 小时前
分享一套锋哥原创的SpringBoot4+Vue3宠物领养网站系统
java
考虑考虑16 小时前
Java实现hmacsha1加密算法
java·后端·java ee
掉鱼的猫17 小时前
Spring Boot → Solon 注解迁移实战指南:一张对照表说清楚
java·spring boot
plainGeekDev17 小时前
广播接收器 → Flow + Lifecycle
android·java·kotlin
plainGeekDev17 小时前
EventBus → SharedFlow
android·java·kotlin