Spring Boot 中是使用 JDK Proxy 动态代理还是 CGLib ?

1. Spring Boot 中是使用 JDK Proxy 动态代理还是 CGLib ?

首先我们知道在 Spring 里面,动态代理的底层,默认使用的是 JDK Proxy 代理,这一点在官方文档有具体的说明:

既然 Spring 中的动态代理默认使用 JDK Proxy,那么在 Spring Boot 中是否也是使用 JDK Proxy 呢 ?

答案是否,Spring Boot 的动态代理默认使用的是 CGLib:

这一点在 spring-configuration-metadata.json 文件中有体现 >>

我们带着疑惑来看下面这个案例,看看 Spring Boot 中的动态代理是否真的使用了 CGLib ?

Controller:

java 复制代码
@RestController
@RequestMapping("/user")
public class UserController {

    @Autowired
    private IUserService userService;

    @RequestMapping("/sayhi")
    public void sayHi() {
        userService.sayHi();
    }
}

service 接口:

java 复制代码
public interface IUserService {
    void sayHi();
}

ServiceImpl:

java 复制代码
@Service
public class UserServiceImpl implements IUserService {
    @Override
    public void sayHi() {
    }
}

使用 AOP 拦截目标方法:

java 复制代码
@Aspect // 切面
@Component
public class LoginAspect {

    @Before("execution(* com.example.demo.service.impl.UserServiceImpl.*(..))")
    public void doBefore(JoinPoint joinPoint) {
        System.out.println("拦截 sayHi 方法:" +
                joinPoint.getSignature().getName());
    }
}

打断点,启动项目访问 sayhi 接口:

此时,我们发现 Spring Boot 动态代理底层确实使用了 CGLib 来实现。

🍁【思考一】

我们都知道使用 CGLIb 实现代理,被代理类是不能被 final 修饰的,因为 CGLib 是通过生成目标类的子类来保证代理类和被代理类的方法名一致,从而实现动态代理的,那如果上述被代理类使用 final 修饰,再次运行程序时,它底层是改成 JDK Proxy 代理呢 ?还是程序执行报错呢 ?

java 复制代码
// 被代理类
@Service
public final class UserServiceImpl implements IUserService {
    @Override
    public void sayHi() {
    }
}

**运行程序时报错:**错误信息也很明显

【结论一】

Spring Boot 使用动态代理时,底层默认使用的是 CGLib,并且是强制性的,一旦目标类被 final 修饰了,那么程序运行时就会报错,而不是使用 JDK Proxy 来完成代理。

如果想要让 Spring Boot 动态代理使用 JDK Proxy 也是可以的,不过需要配置两个属性:

XML 复制代码
# 是否启用 AOP
Spring.aop.auto=true
# 代理方式:设置为 true 表示强制使用 CGLib,false 表示使用 JDK
Spring.aop.proxy-target-class=false

properties 文件中配置了这两属性,再次打断点查看(确实使用了 JDK):

此时就算被代理类使用 final 修饰了,也可以被 JDK 代理。

🍁【思考二】

我们都知道 JDK Proxy 实现的动态代理,被代理类是需要实现一个接口来保证代理类和被代理类的方法名一致,从而实现动态代理的,假设使用 JDK Proxy 动态代理时,目标类没有实现接口,程序运行时,是报错呢 ? 还是使用 CGLib 实现代理 ?

java 复制代码
// 被代理类
@Service
public class UserServiceImpl {
    public void sayHi() {
    }
}

此时,在 Controller 中注入 UserServiceImpl,打断点查看(这次使用了 CGLib):

【最终结论】

1.Spring Boot 默认的动态代理是 CGLib,并且是强制性的,加上 final,项目启动就会报错;

2.想让 Spring Boot 使用 JDK 代理,需要在配置文件中开启 AOP,并把代理方式的属性设为 false,才能使用 JDK 代理;

3.Spring Boot 使用 JDK 代理时,目标类如果没有实现接口,程序运行时,会使用 CGLib 来实现代理。

相关推荐
Unstoppable2230 分钟前
代码随想录算法训练营第 56 天 | 拓扑排序精讲、Dijkstra(朴素版)精讲
java·数据结构·算法·
qinyia1 小时前
WisdomSSH解决docker run命令中log-opt参数不支持导致的容器创建失败问题
java·docker·eureka
电饭叔1 小时前
不含Luhn算法《python语言程序设计》2018版--第8章14题利用字符串输入作为一个信用卡号之二(识别卡号有效)
java·python·算法
小付爱coding1 小时前
Claude Code安装教程【windows版本】
java·git·python
**蓝桉**1 小时前
数组的执行原理,java程序的执行原理
java·开发语言
YDS8292 小时前
MyBatis-Plus精讲 —— 从快速入门到项目实战
java·后端·spring·mybatis·mybatis-plus
库库林_沙琪马2 小时前
7、集成MyBatis
spring boot·mybatis
BBB努力学习程序设计2 小时前
Java条件判断:程序的"决策大脑"
java
我是华为OD~HR~栗栗呀2 小时前
华为OD-C面经-23届学院哦
java·c++·python·华为od·华为·面试
小马爱打代码2 小时前
Spring AI:文生图:调用通义万相 AI 大模型
java·人工智能·spring