Spring 如何使用“三級緩存”來解決“ 循環(huán)依賴”斯撮?
首先來了解一下什么是循環(huán)依賴
@Component
public class A {
@Autowired
B b;
}
@Component
public class B {
@Autowired
A a;
}
在對象A創(chuàng)建過程中,需要注入B扶叉,因為容器中沒有B勿锅,則去創(chuàng)建B勇哗,B創(chuàng)建過程中又需要注入A爽彤,而A在等待B的創(chuàng)建,B在等待A的創(chuàng)建信姓,導致兩者都無法創(chuàng)建成功达吞,無法加入到單例池供用戶使用张弛。
Spring則通過三級緩存來解決循環(huán)依賴的問題,另外如果對象的作用范圍是Prototype酪劫,則無法通過三級緩存解決循環(huán)依賴吞鸭,會拋出BeanCurrentlyInCreationException異常,構造注入的方式覆糟,也無法解決循環(huán)依賴刻剥,只有set注入可以解決。
那么三級緩存又是什么呢滩字?
三級緩存就是三個Map
public class DefaultSingletonBeanRegistry extends SimpleAliasRegistry implements SingletonBeanRegistry {
//一級緩存(單例池造虏,經(jīng)過完成生命周期的對象會放入其中)
private final Map<String, Object> singletonObjects = new ConcurrentHashMap<>(256);
//二級緩存(剛實例化還未初始化的原始對象會放入其中)
private final Map<String, Object> earlySingletonObjects = new HashMap<>(16);
//三級緩存(存放創(chuàng)建某個對象的工廠)
private final Map<String, ObjectFactory<?>> singletonFactories = new HashMap<>(16);
Spring Bean對象從創(chuàng)建到初始化大致會經(jīng)過四個流程
getSingleton()盯滚、doCreateBean()、populateBean()酗电、addSingleton()
- getSingleton:從單例池中獲取bean對象魄藕,如果沒有,則進行創(chuàng)建
- doCreateBean():創(chuàng)建bean對象
- populateBean():填充依賴撵术,如果被填充的對象不存在于單例池背率,則進行創(chuàng)建等四個流程
- addSingleton():將初始化完成的對象加入到單例池
循環(huán)依賴的對象在三級緩存中的遷移過程
- A 創(chuàng)建過程中需要 B, 于是 A 將自己放到三級緩存里面嫩与,去實例化 B
- B 實例化的時候發(fā)現(xiàn)需要 A寝姿,于是 B 先查一級緩存,沒有划滋,再查二級緩存饵筑,還是沒有,再查三級緩存找到了A处坪,然后把三級緩存中的 A 放到二級緩存根资,并刪除三級緩存中的 A
- B 順利初始化完畢,將自己放到一級緩存中(此時 B 中的 A 還是創(chuàng)建中狀態(tài)同窘,并沒有完全初始化)玄帕,刪除三級緩存中的 B然后接著回來創(chuàng)建 A,此時 B 已經(jīng)完成創(chuàng)建想邦,直接從一級緩存中拿到 B裤纹,完成 A 的創(chuàng)建,并將 A 添加到單例池丧没,刪除二級緩存中的 A
圖示: