Spring如何解决循环依赖

1、什么是循环依赖

  多个bean之间相互依赖,造成了一个闭环。 好比:A依赖于B、B依赖于c、c依赖于Aspring

  一般来讲,若是问spring容器内部如何解决循环依赖, 必定是指默认的单例Bean中,属性互相引用的场景。也就是说,Spring的循环依赖,是Spring容器注入时候出现的问题。缓存

    

2、Spring如何解决循环依赖

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

  

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

2,Spring中Bean的生命周期

      

     

3,Bean初始化主要方法

  

  • getSingleton:但愿从容器里面得到单例的bean,没有的话
  • doCreateBean: 没有就建立bean
  • populateBean: 建立完了之后,要填充属性
  • addSingleton: 填充完了之后,再添加到容器进行使用

4,具体说明

  • A建立过程当中须要B,因而A将本身放到三级缓存里面,去实例化B
  • B实例化的时候发现须要A,因而B先查一级缓存,没有,再查二级缓存,仍是没有,再查三级缓存,找到了A而后把三级缓存里面的这个A放到二级缓存里面,并删除三级缓存里面的A
  • B顺利初始化完毕,将本身放到一级缓存里面(此时B里面的A依然是建立中状态)而后回来接着建立A,此时B已经建立结束,直接从一级缓存里面拿到B,而后完成建立,并将A放到一级缓存中。

5,图解

  

3、为何使用三级缓存

1,使用一级缓存

  实例化A -> 将半成品的A放入singletonObjects中->填充A的属性时发现取不到B->实例化B->从singletonObjects中取出A填充B的属性->将成品B放入singletonObjects->将B填充到A的属性中->将成品A放入singletonObjects。安全

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

2,使用二级缓存

a)使用singletonObjects和earlySingletonObjects

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

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

  问题:这样的流程是线程安全的,不过若是A上加个切面(AOP),这种作法就无法知足需求了,由于earlySingletonObjects中存放的都是原始对象,而咱们须要注入的实际上是A的代理对象代理

b)使用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的对象工厂。blog

  问题:这样的流程也适用于普通的IOC以及有并发的场景,但若是A上加个切面(AOP)的话,这种状况也没法知足需求生命周期