Spring如何使用三級緩存解決循環依賴


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

圖示:


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM