Spring解决循环依赖

一、什么是循环依赖

多个Bean之间相互依赖,形成一个闭环。

比如:A依赖于B,B依赖于C,C依赖于A。

通常来说,如果问Spring容器内部如何解决循环依赖,一定是指默认的单列Bean中,属性互相引用的场景。也就是说,Spring的循环依赖,是Spring容器注入时候出现的问题。

二、Spring如何解决循环依赖

 1.Spring中单例Bean的三级缓存

  • 第一级缓存:(单例池) singletonObjects,存放已经经历了完整生命周期的Bean对象
  • 第二级缓存:earlysSingletonObjects,存放早期暴露出来的Bean对象,Bean的生命周期未结束(属性还未填充完整)
  • 第三级缓存:Map<String, ObjectFactory<?>> singletonFactories,存放可以生成Bean的工厂

2.Spring中Bean的生命周期

  • Bean的实例化
  • Bean的属性注入
  • Bean的初始化
  • Bean的销毁

3.Bean初始化的主要方法

  • getSingleton:希望从容器里面获取单例的Bean
  • doCreateBean:没有就创建
  • populateBean:创建完了以后,要填充属性
  • addSingleton:填充完了以后,再添加到容器进行使用 

4.图解 

 5.具体说明

  • A创建过程中需要B,于是A将自己放到三级缓存里,是实例化B;
  • B实例化的时候发现需要A,于是B先查一级缓存,没有,再查二级缓存,还是没有,再查三级缓存,找到了A然后把三级缓存里面的这个A放到二级缓存里面,并删除三级缓存里面的A;
  • B顺利初始化完毕,将自己放到一级缓存里面(此时B里面的A依然是创建中状态)然后回来接着创建A,此时B已经创建结束,直接从一级缓存里面拿到B,然后完成创建,并将A放到一级缓存中。
  • 流程:实例化A -> 创建A的对象工厂并放入singletonFactories中 -> 填充A的属性时发现取不到B -> 实例化B -> B实例化的时候发现需要A,于是B先查一级缓存,没有,再查二级缓存,还是没有,再查三级缓存,找到了从singletonFactories中获取A的对象工厂并获取A -> 将半成品的A放入earlySingletonObjects中 -> 填充到B -> 将成品B放入singletonObjects -> 将B填充到A的属性中 -> 将成品A放入singletonObjects。

三、为什么使用三级缓存

 1.使用一级缓存

流程:实例化A ~> 将半成品的A放入singletonObjects中 ~> 填充A的属性是时发现取不到B ~> 实例化B ~> 从singletonObjects中取出A填充B的属性 ~> 将成品B放入singletonObjects ~> 将B填充到A的属性中 ~> 将成品A 放入singletonObjects。

问题:这种基本流程是通的,但是如果在整个流程进行中,有另一个现场要来取A,那么有可能拿到的只是一个属性都为null的半成品A,这样就会有问题。

2.使用二级缓存

2.1 使用singletonObjects和earlySingletonObjects

成品放在singletonObjects中,半成品放在earlySingletonObjects中

流程:实例化A  ~> 将半成品的A放入earlySingletonObjects中  ~> 填充A的属性时发现取不到B  ~> 实例化B ~> 将半成品的B放入earlySingletonObjects中  ~> 从earlySingletonObjects中取出A填充B的属性 ~> 将成品B放入singletonObjects,并从earlySingletonObjects中删除B  ~> 将B填充到A的属性中  ~> 将成品A放入singletonObjects并删除earlySingletonObjects。

问题:这样的流程是线程安全的,不过如果A上加个切面(AOP),这种做法就没有满足需求了,因为earlySingletonObjects中存放的是原始对象,而我们需要注入的其实是A的代理对象。

2.2 使用singletonObjects和singletonFactories

成品放在singletonObjects中,半成品通过singletonFactories来获取

流程:实例化A ~> 创建A的对象工厂并放入singletonFactories中 ~> 填充A的属性时发现取不到B ~> 实例化B ~> 创建B的对象工厂并放入singletonFactories中 ~> 从singletonFactories中获取A的对象工厂并获取A填充到B ~> 将成品B放入singletonObjects,并从singletonFactories删除B的对象工厂 ~> 将B填充到A的属性中 ~> 将成品A放入singletonObjects并删除A的对象工厂。

问题:这样的流程也适用于普通的IOC以及有并发的场景,但如果A上加个切面(AOP),这种做法就没有满足需求了。

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值