Spring循環(huán)依賴的解決方法詳解
說明:spring如何解決循環(huán)依賴,是面試中經(jīng)常問到的題目,今天我們就來分享一下spring是如何解決循環(huán)依賴問題的。
什么是循環(huán)依賴:
我們先來看看官方文檔的說法:

通俗來講,就是A依賴B或者B依賴A,或者C依賴自己本身,或是三個以上,例如A依賴B,B依賴C,C又依賴A。如下圖:

Spring實例Bean的本質(zhì)
Spring在實例化一個bean的時候,是首先遞歸的實例化其所依賴的所有bean,直到某個bean沒有依賴其他bean,此時就會將該實例返回,然后反遞歸的將獲取到的bean設(shè)置為各個上層bean的屬性的。
循環(huán)依賴主要場景

什么情況下循環(huán)依賴可以被解決
Spring解決循環(huán)依賴是有前置條件的
- 出現(xiàn)循環(huán)依賴的Bean必須要是單例(singleton),如果依賴prototype則完全不會有此需求。
- 依賴注入的方式不能全是構(gòu)造器注入的方式。

解決方式
Spring是通過三級緩存來解決上述問題的:
一級緩存: singletonObjects存儲的是所有創(chuàng)建好了的單例Bean
二級緩存:earlySingletonObjects完成實例化,但是還未進行屬性注入及初始化的對象
三級緩存:singletonFactories提前暴露的一個單例工廠,二級緩存中存儲的就是從這個工廠中獲取到的對象。
三級緩存解決循環(huán)依賴流程:
- 獲取A時首先會嘗試從一級緩存singletonObjects中獲?。?/li>
- 獲取不到就再從二級緩存earlySingletonObjects中獲?。?/li>
- 若是還沒有則嘗試從三級緩存singletonFactories獲?。?/li>
- 還是沒有獲取到則再嘗試創(chuàng)建A對象
- 會執(zhí)行doGetBean->createBean->createBeanInstance并使用構(gòu)造器實例化
- 在嘗試給A進行初始化時,由于B不存在無法完成初始化,則將半成品A放入第二級緩存中,進入B的創(chuàng)建流程。
- 與先前過程相似,在第三級緩存中放入beanName和表達式sharedInstance,進入B的初始化過程
- 由于在第二級緩存中可以找到A,則B可以完成初始化,將成品Bean放入一級緩存中備用,刪除三級緩存中的B
- 同時完成A的初始化,并刪除二級緩存中的半成品A
具體流程圖如下:

最后我們來個小小的總結(jié):
Spring通過三級緩存解決了循環(huán)依賴,其中一級緩存為單例池(singletonObjects),二級緩存為早期曝光對象earlySingletonObjects,三級緩存為早期曝光對象工廠(singletonFactories)。
當(dāng)A、B兩個類發(fā)生循環(huán)引用時,在A完成實例化后,就使用實例化后的對象去創(chuàng)建一個對象工廠,添加到三級緩存中,如果A被AOP代理,那么通過這個工廠獲取到的就是A代理后的對象,如果A沒有被AOP代理,那么這個工廠獲取到的就是A實例化的對象。
當(dāng)A進行屬性注入時,會去創(chuàng)建B,同時B又依賴了A,所以創(chuàng)建B的同時又會去調(diào)用getBean(a)來獲取需要的依賴,此時的getBean(a)會從緩存中獲取。
- 先獲取到三級緩存中的工廠;
- 調(diào)用對象工工廠的getObject方法來獲取到對應(yīng)的對象,得到這個對象后將其注入到B中。緊接著B會走完它的生命周期流程,包括初始化、后置處理器等。
- 當(dāng)B創(chuàng)建完后,會將B再注入到A中,此時A再完成它的整個生命周期。至此,循環(huán)依賴結(jié)束!
到此這篇關(guān)于Spring循環(huán)依賴的解決方法詳解的文章就介紹到這了,更多相關(guān)Spring循環(huán)依賴內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
解決使用@Value(${×××))從properties文件取值的坑
這篇文章主要介紹了解決使用@Value(${×××))從properties文件取值的坑,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教2021-07-07
Java在PowerPoint幻燈片中創(chuàng)建散點圖的方法
散點圖是通過兩組數(shù)據(jù)構(gòu)成多個坐標(biāo)點,考察坐標(biāo)點的分布,判斷兩變量之間是否存在某種關(guān)聯(lián)或總結(jié)坐標(biāo)點的分布模式,這篇文章主要介紹了Java如何在PowerPoint幻燈片中創(chuàng)建散點圖,需要的朋友可以參考下2023-04-04
Java實現(xiàn)為Word每一頁設(shè)置不同圖片水印的效果
Word中設(shè)置水印時,可加載圖片設(shè)置為水印效果,但通常添加水印效果時,會對所有頁面都設(shè)置成統(tǒng)一效果。所以本文為大家介紹了一個方法,可以實現(xiàn)對每一頁或者某個頁面設(shè)置不同的水印效果,需要的可以參考一下2022-02-02

