更多请关注:https://t.zsxq.com/fhroW
文章目录
什么是循环依赖?
循环依赖是spring中的问题,普通的java项目不会有循环依赖。Spring中出现的循环依赖有三种情况:自我依赖、循环依赖、多组依赖
什么是循环依赖?
循环依赖是spring中的问题,普通的java项目不会有循环依赖。
// A依赖了B
class A{
public B b;
}
// B依赖了A
class B{
public A a;
}
以上代码的逻辑就会产生循环依赖,创建A时需要依赖B,创建B时又需要A,陷入了循环。
Spring中出现的循环依赖有三种情况:自我依赖、循环依赖、多组依赖。
解决思想
解决循环依赖的关键在于打破循环。如何打破循环呢?
spring中将A的半成品对象存入缓存(map)中,创建B时,如果没有A的成品对象,就使用A的半成品对象。以此来打破循环。
B依赖的半成品对象会在A创建完成后变成成品对象,因为指向同一块存储空间。
spring如何解决循环依赖
- 为什么需要三级缓存
Spring中使用三级缓存来解决循环依赖。按照上文的思想,解决循环依赖只需要一个缓存即可,为什么Spring要使用三级缓存?
- 为了遵从Spring的设计思想
- 为了代码逻辑清晰
如果只使用一个缓存,对象实例化后就把未进行属性填充半成品对象存入缓存中。这样设计有两个问题:
- 这个缓存中既有半成品对象,也有成品对象,容易造成空指针,代码逻辑不清晰
- 如果对象进行了AOP,那么用户最终得到的是代理对象,但是缓存中存的是普通对象
解决第一个问题:
再加一个缓存,分别存储成品对象和半成品对象。
存储成品对象即Spring中的第一级缓存: singletonObjects
存储半成品对象即Spring中的第二级缓存:earlySingletonObjects
解决第二个问题:
提前进行AOP,AOP本来是在初始化后中进行的,如果出现了循环依赖,将AOP过程提前到实例化后。这样在实例 化后得到的半成品对象可能是普通对象也可能是代理对象。
实例化后处理AOP的这一段逻辑额也存在缓存中,即Spring中的第三级缓存:singletonFactoies
(可能因为存的是生成半成品对象的方法,所以名字以factory结尾)
为什么不把所有的Bean都在实例化后进行AOP?
因为这违背了Spring先创建所有的Bean,再进行初始化的设计思想。
- 使用三级缓存的流程
Spring中使用三级缓存解决循环依赖的流程:
在实例化后将一个方法引用(getEarlyBeanReference())存入第三级缓存,执行这个方法可以获得一个半成品对象,可能是普通对象,也可能是普通对象。
在填充属性时,一次从第一级、第二级、第三级缓存中获取对象,进行填充。
从第三级缓存中获取到半成品对象后,会将该对象从第三级缓存中移除,将半成品对象添加到第二级缓存,下次找的时候就能从第二级缓存中拿到了。
对象创建完成之后会注册对象,即将对象从第二级缓存中移除,添加到第一级缓存中。下次从第一级缓存就直接拿到了。
三级缓存
- 三级缓存分别是什么时候存入的?
实例化后存入第三级缓存,从第三级缓存中获取到半成品对象后,存入第二级缓存,对象创建完成后存入第一级缓存。 - 三级缓存分别是什么时候使用的?
填充属性时依次从三级缓存中寻找对应的Bean - 三级缓存分别存的是什么?
- 一级缓存:即单例池,存储成品对象
- 二级缓存:存储半成品对象
- 三级缓存:存储的是获取半成品的方法引用。可以说存的是工厂对象
- 三级缓存的意义?
三个缓存的意义都是打破循环,解决循环依赖。为了逻辑清晰增加了第二级循环,为了处理AOP且不违背设计思想,增加了第三级循环。
所以如果要说三级缓存各级缓存的意义,我觉得可以说:第三级缓存打破了循环,第二级循环和第一级循环使代码逻辑清晰。
补充
- 判断是否出现了循环依赖?
在创建Bean时,将这个Bean存入一个Map中,表示正在创建,创建结束后移除。其他Bean在创建时,如果发现依赖了这个对象,且这个对象存在于Map中,则说明出现了循环依赖。