什么是循环依赖?
多个bean之间相互依赖,形成了一个闭环。比如A依赖B,B依赖于C,C又依赖于A。
如何解决循环依赖?
一、Spring中单例Bean的三级缓存
-
第一级缓存〈也叫单例池)singletonObjects 存放已经经历了完整生命周期的Bean对象
-
第二级缓存: earlySingletonObjects 存放早期暴露出来的Bean对象,Bean的生命周期未结束(属性还未填充完整)
-
第三级缓存: Map<String, ObiectFactory<?>> singletonFactories 存放可以生成Bean的工厂
二、Spring中Bean的生命周期
-
实例化:执行了bean的构造方法,bean中依赖的对象还未赋值
-
设置属性:给bean中依赖的对象赋值,若被依赖的对象尚未初始化,则先进行该对象的生命周期(递归)。
-
初始化:执行bean的初始化方法,回调方法等。
三、Bean初始化主要方法
-
getSingleton:希望从容器里面获得单例的bean,没有的话
-
doCreateBean: 没有就创建
-
beanpopulateBean: 创建完了以后,要填充属性
-
addSingleton: 填充完了以后,再添加到容器进行使用
四、具体解决讲解
-
A创建过程中需要B,于是A将自己放到三级缓存里面,去实例化B。
-
B实例化的时候发现需要A,于是B先查一级缓存,没有,再查二级缓存,还是没有,再查三级缓存,找到了A。然后把三级缓存里面的这个A放到二级缓存里面,并删除三级缓存里面的A。
-
B顺利初始化完毕,将自己放到一级缓存里面(此时B里面的A依然是创建中状态)然后回来接着创建A,此时B已经创建结束,直接从一级缓存里面拿到B,然后完成创建,并将A放到一级缓存中。
图解: 假设A、B形成循环依赖