为什么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
执行代理方法(CGLIBhello1hello1hello16) -
然后代理方法(CGLIBhello1hello1hello16)才通过super关键字调用被代理类的hello1()方法
3. SpringBoot中是如何创建代理对象的
Spring中有一个BeanPostProcessor
类,这是一个钩子类,主要就负责在Bean的各个生命周期阶段执行回调方法
其中就有一个子类:AnnotationAwareAspectJAutoProxyCreator
,这个类正是负责创建代理对象的
紧接着我们来看这个子类的结构图:
还是比较的复杂的,我们不管,我们就看BeanPostProcessor
的postProcessAfterInitialization
在子类中是如何实现的
这个方法其实很简单,就是看bean是否需要被代理,如果需要就将代理后的对象返回
,换句话说注册到Spring容器中的是代理对象
, 并且代理对象是没有经过依赖注入的
紧接着我们看在SpringMVC中当通过参数解析器解析出入参的时候,是怎么执行目标方法的
这里是通过getBean()获取的对象是什么,就是我们代理对象
前面已经介绍过这个是如何正确的执行目标方法的,就是将原方法进行重写,最终在MethodInterceptor
中才会执行
CGLIB$hello1$6()
方法,而CGLIB$hello1$6()
方法才会通过super
关键字调用父类,也就是进行过依赖注入的对象
可是如果我们的方法是用private
修饰的呢,private
和final
修饰的关键字是不能被重写的,也就是在代理对象中不存在我刚才说的CGLIB$hello1$6()
方法,那这个时候反射执行的方法是在哪里执行的呢,就是在当前对象也就是代理对象中执行的,代理对象又没有进行依赖注入
,所以说使用@Resource
或者@Autowired
一定为空
但即使在这种情况下我们依旧可以换一种思路去调用被代理对象中的属性:
4. 总结
总结:
private
关键字并不会导致动态代理失效- 只不过由于注册到容器中的是代理对象,而代理对象本身无法重写
private
和final
修饰的方法,又由于Spring没有对代理对象中的属性进行依赖注入 - 最终才导致执行方法的时候只能调用代理对象中的属性,而无法调用父类(被代理对象)的属性
转载自:https://juejin.cn/post/7280165108222378025