Spring源码系列五:循环依赖

更多请关注: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如何解决循环依赖

  1. 为什么需要三级缓存
    Spring中使用三级缓存来解决循环依赖。按照上文的思想,解决循环依赖只需要一个缓存即可,为什么Spring要使用三级缓存?
  • 为了遵从Spring的设计思想
  • 为了代码逻辑清晰

如果只使用一个缓存,对象实例化后就把未进行属性填充半成品对象存入缓存中。这样设计有两个问题:

  1. 这个缓存中既有半成品对象,也有成品对象,容易造成空指针,代码逻辑不清晰
  2. 如果对象进行了AOP,那么用户最终得到的是代理对象,但是缓存中存的是普通对象

解决第一个问题
再加一个缓存,分别存储成品对象和半成品对象。
存储成品对象即Spring中的第一级缓存: singletonObjects
存储半成品对象即Spring中的第二级缓存:earlySingletonObjects

解决第二个问题
提前进行AOP,AOP本来是在初始化后中进行的,如果出现了循环依赖,将AOP过程提前到实例化后。这样在实例 化后得到的半成品对象可能是普通对象也可能是代理对象。
实例化后处理AOP的这一段逻辑额也存在缓存中,即Spring中的第三级缓存:singletonFactoies
(可能因为存的是生成半成品对象的方法,所以名字以factory结尾)

为什么不把所有的Bean都在实例化后进行AOP?
因为这违背了Spring先创建所有的Bean,再进行初始化的设计思想。

  1. 使用三级缓存的流程
    Spring中使用三级缓存解决循环依赖的流程:
    在这里插入图片描述

在实例化后将一个方法引用(getEarlyBeanReference())存入第三级缓存,执行这个方法可以获得一个半成品对象,可能是普通对象,也可能是普通对象。
在填充属性时,一次从第一级、第二级、第三级缓存中获取对象,进行填充。

从第三级缓存中获取到半成品对象后,会将该对象从第三级缓存中移除,将半成品对象添加到第二级缓存,下次找的时候就能从第二级缓存中拿到了。
对象创建完成之后会注册对象,即将对象从第二级缓存中移除,添加到第一级缓存中。下次从第一级缓存就直接拿到了。

三级缓存

  1. 三级缓存分别是什么时候存入的?
    实例化后存入第三级缓存,从第三级缓存中获取到半成品对象后,存入第二级缓存,对象创建完成后存入第一级缓存。
  2. 三级缓存分别是什么时候使用的?
    填充属性时依次从三级缓存中寻找对应的Bean
  3. 三级缓存分别存的是什么?
  • 一级缓存:即单例池,存储成品对象
  • 二级缓存:存储半成品对象
  • 三级缓存:存储的是获取半成品的方法引用。可以说存的是工厂对象
  1. 三级缓存的意义?
    三个缓存的意义都是打破循环,解决循环依赖。为了逻辑清晰增加了第二级循环,为了处理AOP且不违背设计思想,增加了第三级循环。
    所以如果要说三级缓存各级缓存的意义,我觉得可以说:第三级缓存打破了循环,第二级循环和第一级循环使代码逻辑清晰。

补充

  1. 判断是否出现了循环依赖?
    在创建Bean时,将这个Bean存入一个Map中,表示正在创建,创建结束后移除。其他Bean在创建时,如果发现依赖了这个对象,且这个对象存在于Map中,则说明出现了循环依赖。
  • 12
    点赞
  • 7
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值