Spring如何使用三級緩存解決循環依賴
首先來了解一下什么是循環依賴
@Component
public class A {
@Autowired
B b;
}
@Component
public class B {
@Autowired
A a;
}
在對象A創建過程中,需要注入B,因為容器中沒有B,則去創建B,B創建過程中又需要注入A,而A在等待B的創建,B在等待A的創建,導致兩者都無法創建成功,無法加入到單例池供用戶使用。
Spring則通過三級緩存來解決循環依賴的問題,另外如果對象的作用范圍是Prototype,則無法通過三級緩存解決循環依賴,會拋出BeanCurrentlyInCreationException
異常,構造注入的方式,也無法解決循環依賴,只有set注入可以解決。
那么三級緩存又是什么呢?
三級緩存就是三個Map
public class DefaultSingletonBeanRegistry extends SimpleAliasRegistry implements SingletonBeanRegistry {
//一級緩存(單例池,經過完成生命周期的對象會放入其中)
private final Map<String, Object> singletonObjects = new ConcurrentHashMap<>(256);
//二級緩存(剛實例化還未初始化的原始對象會放入其中)
private final Map<String, Object> earlySingletonObjects = new HashMap<>(16);
//三級緩存(存放創建某個對象的工廠)
private final Map<String, ObjectFactory<?>> singletonFactories = new HashMap<>(16);
Spring Bean對象從創建到初始化大致會經過四個流程
getSingleton()
、doCreateBean()
、populateBean()
、addSingleton()
-
getSingleton
:從單例池中獲取bean對象,如果沒有,則進行創建 -
doCreateBean()
:創建bean對象 -
populateBean()
:填充依賴,如果被填充的對象不存在於單例池,則進行創建等四個流程 -
addSingleton()
:將初始化完成的對象加入到單例池
循環依賴的對象在三級緩存中的遷移過程
-
A 創建過程中需要 B, 於是 A 將自己放到三級緩存里面,去實例化 B
-
B 實例化的時候發現需要 A,於是 B 先查一級緩存,沒有,再查二級緩存,還是沒有,再查三級緩存
找到了A,然后把三級緩存中的 A 放到二級緩存,並刪除三級緩存中的 A
-
B 順利初始化完畢,將自己放到一級緩存中(此時 B 中的 A 還是創建中狀態,並沒有完全初始化),刪除三級緩存中的 B
然后接着回來創建 A,此時 B 已經完成創建,直接從一級緩存中拿到 B,完成 A 的創建,並將 A 添加到單例池,刪除二級緩存中的 A
圖示: