为什么SpringBoot中使用Cglib代理private方法会失效

1. 背景介绍

大家在SpringBoot中使用Cglib动态代理的时候,发现使用private关键字修饰方法会导致代理失效,就像下面这样 网上很多文章的解决方法都是将private改为public就可以了,但很少有文章解释为什么

本文将通过分析SpringBoot中如何创建代理对象的,来解释说明private为什么不行

严格意义上来说private并不会导致代理失效,而只是影响了调用的对象而已

2. Cglib

Cglib是一个强大的、高性能的代码生成库,Cglib是利用ASM字节码处理框架会生成三个类

  • 代理类
  • 代理类对应的FastClass类
  • 被代理类对应的FastClass类

代理类:

  • 继承了被代理类的

  • 内部有一个静态代码块会利用反射将被代理类的所有方法对应的Method保存起来,并根据Method创建MethodProxy也保存起来

  • MethodProxy:主要是保存原始方法和代理方法的签名(Signature)

FastClass类在这里不重要,我们主要看在代理的情况下,Cglib是怎么执行方法的,比方说执行hello1()方法

  • 代理类中的hello1()方法实际上是被代理类重写了的

  • hello1()是先通过MethodInterceptor执行代理方法(CGLIB <math xmlns="http://www.w3.org/1998/Math/MathML"> h e l l o 1 hello1 </math>hello16)

  • 然后代理方法(CGLIB <math xmlns="http://www.w3.org/1998/Math/MathML"> h e l l o 1 hello1 </math>hello16)才通过super关键字调用被代理类的hello1()方法

3. SpringBoot中是如何创建代理对象的

Spring中有一个BeanPostProcessor类,这是一个钩子类,主要就负责在Bean的各个生命周期阶段执行回调方法

其中就有一个子类:AnnotationAwareAspectJAutoProxyCreator,这个类正是负责创建代理对象的

紧接着我们来看这个子类的结构图:

还是比较的复杂的,我们不管,我们就看BeanPostProcessorpostProcessAfterInitialization在子类中是如何实现的

这个方法其实很简单,就是看bean是否需要被代理,如果需要就将代理后的对象返回,换句话说注册到Spring容器中的是代理对象, 并且代理对象是没有经过依赖注入的

紧接着我们看在SpringMVC中当通过参数解析器解析出入参的时候,是怎么执行目标方法的

这里是通过getBean()获取的对象是什么,就是我们代理对象

前面已经介绍过这个是如何正确的执行目标方法的,就是将原方法进行重写,最终在MethodInterceptor中才会执行 CGLIB$hello1$6()方法,而CGLIB$hello1$6()方法才会通过super关键字调用父类,也就是进行过依赖注入的对象

可是如果我们的方法是用private修饰的呢,privatefinal修饰的关键字是不能被重写的,也就是在代理对象中不存在我刚才说的CGLIB$hello1$6()方法,那这个时候反射执行的方法是在哪里执行的呢,就是在当前对象也就是代理对象中执行的,代理对象又没有进行依赖注入,所以说使用@Resource或者@Autowired一定为空

但即使在这种情况下我们依旧可以换一种思路去调用被代理对象中的属性:

4. 总结

总结:

  • private关键字并不会导致动态代理失效
  • 只不过由于注册到容器中的是代理对象,而代理对象本身无法重写privatefinal修饰的方法,又由于Spring没有对代理对象中的属性进行依赖注入
  • 最终才导致执行方法的时候只能调用代理对象中的属性,而无法调用父类(被代理对象)的属性
相关推荐
懋学的前端攻城狮23 分钟前
Vue源码解析-01:从创建到挂载的完整流程
前端·vue.js·源码
崔lc1 小时前
Springboot项目集成Ai模型(阿里云百炼-DeepSeek)
java·spring boot·后端·ai
耀耀_很无聊1 小时前
03_跨域问题解决
java·spring boot·跨域·satoken
寒山李白2 小时前
Spring Boot面试题精选汇总
java·spring boot·后端·面试
磊叔的技术博客2 小时前
随笔小记:SpringBoot 3 集成 SpringDoc OpenAPI
spring boot·后端
计算机毕设定制辅导-无忧学长3 小时前
Spring Boot 与 Kafka 的深度集成实践(一)
spring boot·kafka·linq
crud3 小时前
Spring Boot 使用 @Async 实现异步操作:从入门到实战,一文讲透
java·spring boot
工业互联网专业4 小时前
基于django+vue的健身房管理系统-vue
vue.js·python·django·毕业设计·源码·课程设计·健身房管理系统
迢迢星万里灬4 小时前
Java求职者面试指南:Spring、Spring Boot、MyBatis框架与计算机基础问题解析
java·spring boot·spring·mybatis·计算机基础·面试指南
风象南5 小时前
SpringBoot的5种签到打卡实现方案
java·spring boot·后端