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

SpringIOC DI循環(huán)依賴實(shí)例詳解

 更新時(shí)間:2020年03月06日 13:35:23   作者:FFStayF  
這篇文章主要介紹了SpringIOC——DI循環(huán)依賴,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下

要弄清楚循環(huán)依賴

1、需要知道Bean初始化的兩個(gè)階段

① Bean實(shí)例化創(chuàng)建實(shí)例對(duì)象(new Bean())

② Bean實(shí)例對(duì)象初始化(DI:注解自動(dòng)注入)

2、DefaultSingletonBeanRegistry類中的5個(gè)容器

/** 記錄已將創(chuàng)建的單例<beanName,singletonBean> */
  private final Map<String, Object> singletonObjects = new ConcurrentHashMap<>(256);

  /** 記錄singletonFactory<beanName,singletonFactory> singeletonFactory中存放beanName和上面的①階段的bean:Bean實(shí)例化實(shí)例對(duì)象(還未初始化DI)*/
  private final Map<String, ObjectFactory<?>> singletonFactories = new HashMap<>(16);

  /** 記錄早期的singletonBean 存放的也是① */
  private final Map<String, Object> earlySingletonObjects = new HashMap<>(16);

  /** 存放已經(jīng)初始化后的beanName,有序 */
  private final Set<String> registeredSingletons = new LinkedHashSet<>(256);

  /** 記錄正在初始化的bean的beanName */
  private final Set<String> singletonsCurrentlyInCreation = Collections.newSetFromMap(new ConcurrentHashMap<>(16));

其中跟循環(huán)依賴相關(guān)的是singletonFactories、singeletonsCurrentlyInCreation、earlysingletonObjects.

3、循環(huán)依賴實(shí)現(xiàn)

①bean初始化前后會(huì)打標(biāo),加入到singletonsCurrentlyInCreation容器中,這個(gè)打標(biāo)會(huì)在核心方法getSingleton()中起作用

/* org.springframework.beans.factory.support.DefaultSingletonBeanRegistry#getSingleton(java.lang.String, org.springframework.beans.factory.ObjectFactory<?>) */
  public Object getSingleton(String beanName, ObjectFactory<?> singletonFactory) {
    ...
    //循環(huán)依賴相關(guān):初始化前先singletonsCurrentlyInCreation.add(beanName)
    beforeSingletonCreation(beanName);
    ...
    //lamda表達(dá)式:其實(shí)是調(diào)用createBean(beanName, mbd, args):Bean初始化
    singletonObject = singletonFactory.getObject();
    ...
    //循環(huán)依賴相關(guān):初始化后singletonsCurrentlyInCreation.remove(beanName)
    afterSingletonCreation(beanName);
    ...//初始化完后
    //this.singletonObjects.put(beanName, singletonObject);放入到單例容器中
    //this.singletonFactories.remove(beanName);清空循環(huán)依賴的兩個(gè)打標(biāo)
    //this.earlySingletonObjects.remove(beanName);
    //this.registeredSingletons.add(beanName);放入單例beanName容器中
    addSingleton(beanName, singletonObject);
    ...
    }
  }

② 上面singletonObject = singletonFactory.getObject()時(shí)向singletonFactories中記錄了(new Bean()),singletonFactories也會(huì)在核心方法getSingleton()中起作用

/* org.springframework.beans.factory.support.AbstractAutowireCapableBeanFactory#doCreateBean */
  protected Object doCreateBean(final String beanName, final RootBeanDefinition mbd, final @Nullable Object[] args)
      throws BeanCreationException {
    ...
    //循環(huán)依賴相關(guān)邏輯:
    //this.singletonFactories.put(beanName, singletonFactory);
    //將實(shí)例化bean(①階段)、beanName組裝成singletonFactory裝入singletonFactories容器
    //this.earlySingletonObjects.remove(beanName);
    //刪除earlySingletonObjects中beanName
    addSingletonFactory(beanName, () -> getEarlyBeanReference(beanName, mbd, bean));
    ...
    //實(shí)例初始化 就是在這里面實(shí)現(xiàn)依賴注入DI的:反射實(shí)現(xiàn)
    //調(diào)用AutowiredAnnotationBeanPostProcessor.postProcessProperties
    populateBean(beanName, mbd, instanceWrapper);
    ...
  }

③ 核心方法getSingleton

/* org.springframework.beans.factory.support.DefaultSingletonBeanRegistry#getSingleton(java.lang.String, boolean) */
  protected Object getSingleton(String beanName, boolean allowEarlyReference) {
    Object singletonObject = this.singletonObjects.get(beanName);
     //循環(huán)依賴核心就在于這個(gè)判斷,由于打標(biāo)+記錄了①階段的bean,
    //循環(huán)依賴第二次調(diào)用getBean("a")時(shí),這里會(huì)直接返回第一次調(diào)用getBean("a")創(chuàng)建的①階段的bean
    //而不會(huì)調(diào)用createBean("a")再次bean初始化(造成兩個(gè)bean的循環(huán)創(chuàng)建)
    if (singletonObject == null && isSingletonCurrentlyInCreation(beanName)) {
      synchronized (this.singletonObjects) {
        singletonObject = this.earlySingletonObjects.get(beanName);
        if (singletonObject == null && allowEarlyReference) {
          ObjectFactory<?> singletonFactory = this.singletonFactories.get(beanName);
          if (singletonFactory != null) {
            singletonObject = singletonFactory.getObject();
            this.earlySingletonObjects.put(beanName, singletonObject);
            this.singletonFactories.remove(beanName);
          }
        }
      }
    }
    return singletonObject;
  }

④ 循環(huán)依賴流程

/* org.springframework.beans.factory.support.AbstractBeanFactory#doGetBean */
  protected <T> T doGetBean(final String name, @Nullable final Class<T> requiredType,
      @Nullable final Object[] args, boolean typeCheckOnly) throws BeansException {
    ...
    //假設(shè)A、B互相依賴
    //第一次getBean(A),sharedInstance == null,走else,createBean
    //A正在創(chuàng)建打標(biāo),①中beforeSingletonCreation()
    //A實(shí)例化后保存到singletonFactories中②中addSingletonFactory(beanName,singletonFactory)
    //DI依賴注入:②中populateBean(beanName,mbd,instanceWrapper),發(fā)現(xiàn)依賴B,調(diào)用getBean(B)初始化B的單例
    //調(diào)用getBean(B)重復(fù)上面步驟,DI依賴注入發(fā)現(xiàn)依賴A,調(diào)用getBean(A)
    //第二次getBean(A),③中if(singletonObject == null && isSingletonCurrentlyInCreation(A))由于打標(biāo)了所以返回singleFactory.getObject()
    //下面if條件直接返回bean,沒(méi)有走else破壞了循環(huán)
    Object sharedInstance = getSingleton(beanName);
    if (sharedInstance != null && args == null) {
      //
      bean = getObjectForBeanInstance(sharedInstance, name, beanName, null);
    }

    else {
      ...
      //
      createBean(beanName, mbd, args);
      bean = getObjectForBeanInstance(sharedInstance, name, beanName, null);
    }
    return bean;      
  }

四、總結(jié)

未看源碼之前,其實(shí)對(duì)循環(huán)依賴有一個(gè)想法:循環(huán)依賴可以看做是一個(gè)死鎖。

預(yù)防死鎖的方法:打破死鎖的四個(gè)必要條件(互斥、請(qǐng)求并等待、不可剝奪、循環(huán)等待),由于循環(huán)依賴的資源是對(duì)象自身,所以常用破壞循環(huán)等待條件方法:編號(hào)順序執(zhí)行,不適用

選擇破壞請(qǐng)求并等待條件:先創(chuàng)建對(duì)象,再賦值,模型

A a = new A();
B b = new B();
A.b = b;
B.a = a;

研究源碼之后發(fā)現(xiàn):想法差不多,但是代碼實(shí)現(xiàn)非常精彩。模型(打標(biāo)沒(méi)想到過(guò))

A a = new A();
B b = new B();
b.a = a;
a.b = b;

以上就是本文的全部?jī)?nèi)容,希望對(duì)大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。

相關(guān)文章

  • java.Net.UnknownHostException異常處理問(wèn)題解決

    java.Net.UnknownHostException異常處理問(wèn)題解決

    這篇文章主要介紹了java.Net.UnknownHostException異常處理方法,問(wèn)題原因是在系統(tǒng)的?/etc/Hostname中配置了主機(jī)名,而在/etc/hosts文件中沒(méi)有相應(yīng)的配置,本文給大家詳細(xì)講解,需要的朋友可以參考下
    2023-03-03
  • Lombok中@EqualsAndHashCode注解的使用及說(shuō)明

    Lombok中@EqualsAndHashCode注解的使用及說(shuō)明

    這篇文章主要介紹了Lombok中@EqualsAndHashCode注解的使用及說(shuō)明,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教
    2023-03-03
  • Java中HashMap和Hashtable的區(qū)別小結(jié)

    Java中HashMap和Hashtable的區(qū)別小結(jié)

    本文主要介紹了Java中HashMap和Hashtable的區(qū)別小結(jié),文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧
    2023-07-07
  • 關(guān)于springcloud報(bào)錯(cuò)報(bào)UnsatisfiedDependencyException的問(wèn)題

    關(guān)于springcloud報(bào)錯(cuò)報(bào)UnsatisfiedDependencyException的問(wèn)題

    這篇文章主要介紹了關(guān)于springcloud報(bào)錯(cuò)報(bào)UnsatisfiedDependencyException的問(wèn)題,本文給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下
    2020-11-11
  • Java實(shí)現(xiàn)堆排序(大根堆)的示例代碼

    Java實(shí)現(xiàn)堆排序(大根堆)的示例代碼

    這篇文章主要介紹了Java實(shí)現(xiàn)堆排序(大根堆)的示例代碼,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧
    2019-10-10
  • 使用Jitpack發(fā)布開(kāi)源Java庫(kù)的詳細(xì)流程

    使用Jitpack發(fā)布開(kāi)源Java庫(kù)的詳細(xì)流程

    這篇文章主要介紹了使用Jitpack發(fā)布開(kāi)源Java庫(kù)的詳細(xì)流程,本文通過(guò)圖文實(shí)例代碼相結(jié)合給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下
    2022-02-02
  • java如何將控制臺(tái)輸出日志寫入到指定文件中

    java如何將控制臺(tái)輸出日志寫入到指定文件中

    這篇文章主要介紹了java如何將控制臺(tái)輸出日志寫入到指定文件中問(wèn)題,具有很好的參考價(jià)值,希望對(duì)大家有所幫助,如有錯(cuò)誤或未考慮完全的地方,望不吝賜教
    2024-04-04
  • springmvc json類型轉(zhuǎn)換錯(cuò)誤解決方案

    springmvc json類型轉(zhuǎn)換錯(cuò)誤解決方案

    這篇文章主要介紹了springmvc json類型轉(zhuǎn)換錯(cuò)誤解決方案,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下
    2019-12-12
  • 關(guān)于Process的waitFor死鎖問(wèn)題及解決方案

    關(guān)于Process的waitFor死鎖問(wèn)題及解決方案

    這篇文章主要介紹了關(guān)于Process的waitFor死鎖問(wèn)題及解決方案,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教
    2021-12-12
  • java加載properties文件的六種方法總結(jié)

    java加載properties文件的六種方法總結(jié)

    這篇文章主要介紹了java加載properties文件的六種方法總結(jié)的相關(guān)資料,需要的朋友可以參考下
    2017-05-05

最新評(píng)論