Spring的循环依赖
- 什么是循环依赖?
- 循环依赖并不能彻底解决,就算非构造方法注入也不能,为什么?
- "二级缓存"如何解决循环依赖?
- spring为什么用"三级缓存"去解决循环依赖?
- 总结"三级缓存"的精妙之处
构造AService对应的bean的过程:
- 扫描类--->BeanDefinition
- aService = new aService();//原始对象(每个bean创建时都会生成原始对象,并且放在缓存中)
- aService填充属性bservice-----------》BService的Bean------------》构造一个BService---------》
- Aware,init
- BeanPostProcessor:对前面所生成的对象进行加工 这一步会进行AOP----->生成一个代理对象
- 单例池(singleTonObjects,beanName:bean对象)
Aservice--->bean对象--->代理对象(最后单例池放的就是代理对象[不是原对象])
构造AService对应的bean的过程:
- 1. 扫描类--->BeanDefinition
- 2. bService = new BService();//原始对象
- 3. bService填充属性aservice-----------》AService的Bean------------》单例池去找?找不到
- 4. Aware,init
- 5. BeanPostProcessor:对前面所生成的对象进行加工
- 6. 单例池(singleTonObjects,beanName:bean对象)
关于替换容器的对象(使用BeanPostProcessy或AOP):
- BeanPostProcessor:你返回变量是没有返回对应对象的原始对象(就是你替换了接口默认的返回值),会导致bean的注入失败
- AOP:如果你使用了aop那么你getBean()时得到的对象是代理对象,这样为什么不会报值的注入异常-----
Spring存在三级缓存(都是Map):
- 单例池singleTonFactory :已经经历了完整生命周期的bean对象。
- earlySingleTonFactory(二级缓存):比singletonObjects多了一个early,表示缓存的是早期的bean对象。早期是什么意思?表示Bean的生命周期还没走完就把这个Bean放入了earlySingletonObjects。缓存提前拿原始对象进行了AOP之后得到的代理对象,原始对象还没有进行属性注入和后续的BeanPostProcessor等生命周期
- 单例对象工厂(singleTonFactory) :缓存的是一个ObjectFactory,主要用来去生成原始对象进行了AOP之后得到的代理对象,在每个Bean的生成过程中,都会提前暴露一个工厂,这个**工厂可能用到,也可能用不到**,如果没有出现循环依赖依赖本bean,那么这个工厂无用,本bean按照自己的生命周期执行,执行完后直接把本bean放入singletonObjects中即可,如果出现了循环依赖依赖了本bean,则另外那个bean执行ObjectFactory提交得到一个AOP之后的代理对象(如果有AOP的话,如果无需AOP,则直接得到一个原始对象)。
三级缓存处理逻辑的代码:
@Nullable protected Object getSingleton(String beanName, boolean allowEarlyReference) { Object singletonObject = this.singletonObjects.get(beanName); if (singletonObject == null && isSingletonCurrentlyInCreation(beanName)) { synchronized (this.singletonObjects) { //没有earlySingleTonObjects会怎么样 singletonObject = this.earlySingletonObjects.get(beanName); if (singletonObject == null && allowEarlyReference) { //为什么需要singleTonFactories ObjectFactory<?> singletonFactory = this.singletonFactories.get(beanName); if (singletonFactory != null) { /* 就是执行()->getEarlyBeanReference(beanName,mbd,bean原始对象),执行这段lamda表达式就是得到一个代理对象(cglib代理,此时他的原始对象还没有赋值) */ singletonObject = singletonFactory.getObject();//AOP代理 this.earlySingletonObjects.put(beanName, singletonObject); this.singletonFactories.remove(beanName); } } } } return singletonObject; }
第三级缓存(singleTonFactories)如何解决循环依赖的:
singletonFactories----》根据beanName得到一个objectFactory-(就是调用getEarlyBeanReference)----》此时我得到了一个A原始对象经过AOP之后的代理对象---------》将代理对象添加到earlySingletonObject(第二级缓存)------》此时就会将这个添加的代理对象填充到你的bean的属性中-----》最后填充好属性的bean在走一下AOP判断和BeanPostProcess的执行流程-----》放入到单例池
从singletonFactories根据beanName得到一个ObjectFactory,然后执行ObjectFactory,也就是执行getEarlyBeanReference方法,此时会得到一个A原始对象经过AOP之后的代理对象,然后把该代理对象放入earlySingletonObjects中,注意此时并没有把代理对象放入singletonObjects中,那什么时候放入到singletonObjects中呢?
我们这个时候得来理解一下earlySingletonObjects的作用,此时,我们只得到了A原始对象的代理对象,这个对象还不完整,因为A原始对象还没有进行属性填充,所以此时不能直接把A的代理对象放入singletonObjects中,所以只能把代理对象放入earlySingletonObjects,假设现在有其他对象依赖了A,那么则可以从earlySingletonObjects中得到A原始对象的代理对象了,并且是A的同一个代理对象。
当B创建完了之后,A继续进行生命周期,而A在完成属性注入后,会按照它本身的逻辑去进行AOP,而此时我们知道A原始对象已经经历过了AOP,所以对于A本身而言,不会再去进行AOP了,那么怎么判断一个对象是否经历过了AOP呢?会利用上文提到的earlyProxyReferences,在AbstractAutoProxyCreator的postProcessAfterInitialization方法中,会去判断当前beanName是否在earlyProxyReferences,如果在则表示已经提前进行过AOP了,无需再次进行AOP。
对于A而言,进行了AOP的判断后,以及BeanPostProcessor的执行之后,就需要把A对应的对象放入singletonObjects中了,但是我们知道,应该是要A的代理对象放入singletonObjects中,所以此时需要从earlySingletonObjects中得到代理对象,然后入singletonObjects中。