欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

淺談Spring中的循環(huán)依賴問題與解決方案

 更新時間:2023年12月28日 09:58:45   作者:一個風(fēng)輕云淡  
這篇文章主要介紹了淺談Spring中的循環(huán)依賴問題與解決方案,循環(huán)依賴就是兩個或則兩個以上的bean互相持有對方,最終形成閉環(huán),比如A依賴于B,B依賴于C,C又依賴于A,需要的朋友可以參考下

循環(huán)依賴問題

循環(huán)依賴就是兩個或則兩個以上的bean互相持有對方,最終形成閉環(huán)。比如A依賴于B,B依賴于C,C又依賴于A。

在創(chuàng)建A對象的同時需要使用的B對象,在創(chuàng)建B對象的同時需要使用到A對象

循環(huán)依賴可能會導(dǎo)致程序出現(xiàn)各種問題,比如編譯錯誤、運行時錯誤、死鎖等。因此,避免循環(huán)依賴是編寫高質(zhì)量軟件的重要方面之一。

解決方案討論

Spring中循環(huán)依賴場景有:

  • 構(gòu)造器的循環(huán)依賴field屬性的循環(huán)依賴
  • 我們將對這倆個場景進行解決方案的討論

field屬性的循環(huán)依賴解決方案

Spring 通過提前曝光機制,利用三級緩存解決field屬性的循環(huán)依賴。

對應(yīng)的三級緩存如下所示:

//單實例對象注冊器
public class DefaultSingletonBeanRegistry extends SimpleAliasRegistry implements SingletonBeanRegistry {
    private static final int SUPPRESSED_EXCEPTIONS_LIMIT = 100;
    //一級緩存
    private final Map<String, Object> singletonObjects = new ConcurrentHashMap(256);
    //三級緩存
    private final Map<String, ObjectFactory<?>> singletonFactories = new HashMap(16);
    //二級緩存
    private final Map<String, Object> earlySingletonObjects = new ConcurrentHashMap(16);
}
緩存名稱源碼名稱作用
一級緩存singletonObjects單例池,緩存已經(jīng)經(jīng)歷了完整的生命周期,已經(jīng)初始化完成的bean對象
二級緩存earlySingletonObjects緩存早期的bean對象(生命周期還沒走完)
三級緩存singletonFactories緩存的是ObjectFactory,表示對象工廠,用來創(chuàng)建某個對象的 

二級緩存為早期曝光對象earlySingletonObjects,三級緩存為早期曝光對象工廠(singletonFactories)。 

解決過程: 

  • 先實例A對象,同時會創(chuàng)建ObjectFactory對象存入三級緩存singletonFactories
  • A在初始化的時候需要B對象,這個走B的創(chuàng)建的邏輯
  • B實例化完成,也會創(chuàng)建ObjectFactory對象存入三級緩存singletonFactories
  • B需要注入A,通過三級緩存中獲取ObjectFactory來生成一個A的對象同時存入二級緩存,這個是有兩種情況,一個是可能是A的普通對象,另外一個是A的代理對象,都可以讓ObjectFactory來生產(chǎn)對應(yīng)的對象,這也是三級緩存的關(guān)鍵
  • B通過從通過二級緩存earlySingletonObjects 獲得到A的對象后可以正常注入,B創(chuàng)建成功,存入一級緩存singletonObjects
  • 回到A對象初始化,因為B對象已經(jīng)創(chuàng)建完成,則可以直接注入B,A創(chuàng)建成功存入一次緩存singletonObjects
  • 二級緩存中的臨時對象A清除

三級緩存的Value是ObjectFactory,可以從里邊拿到代理對象 

二級緩存存在的必要就是為了性能,從三級緩存的工廠里創(chuàng)建出對象,再扔到二級緩存(這樣就不用每次都要從工廠里拿) 

構(gòu)造器的循環(huán)依賴解決方案

 基于field屬性的循環(huán)依賴,Spring幫我們利用三級緩存自己解決掉了,但對于構(gòu)造器的循環(huán)依賴,spring不能幫我們解決掉,這個時候?qū)τ跇?gòu)造器的循環(huán)依賴問題,我們可以修改代碼把他換成 基于field屬性的循環(huán)依賴。還有一種方法就是使用@Lazy 注解

  public UserService(@Lazy OrderService orderService){
    this.orderService = orderService;
  }

@Lazy注解指示是否要延遲初始化 bean。 它可以用于@Component和@Bean定義。

@Lazy bean 不會被初始化,直到被另一個 bean 引用或從BeanFactory中顯式檢索。 不使用@Lazy注解的 Bean 會被初始化。 

在Java中,@Lazy是一個注解,它可以用來延遲初始化對象的創(chuàng)建。通常情況下,當(dāng)對象被創(chuàng)建時,它的所有屬性都會被初始化。但是,有時候我們可能不想在對象被創(chuàng)建時立即初始化它的所有屬性,特別是當(dāng)某些屬性的初始化需要大量的計算或者需要耗費很多時間時。在這種情況下,我們可以使用@Lazy注解來延遲它們的初始化,直到需要使用它們時才會進行實際的初始化操作。

在構(gòu)造方法上使用@Lazy注解的效果和在屬性上使用@Lazy注解是類似的,只是它會影響整個對象的初始化順序。在使用@Lazy注解時,構(gòu)造方法中不會進行對象屬性的初始化操作,而是在第一次訪問某個被@Lazy注解所標(biāo)記的屬性時進行初始化。這就意味著,當(dāng)我們使用@Lazy注解時,只有在真正需要使用某個屬性時才會進行初始化。這樣就可以避免某些屬性的無效初始化,提高了代碼的性能和效率。

到此這篇關(guān)于淺談Spring中的循環(huán)依賴問題與解決方案的文章就介紹到這了,更多相關(guān)Spring循環(huán)依賴解決方案內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!

相關(guān)文章

最新評論