Spring循環(huán)依賴導(dǎo)致的@PostConstruct順序沒有按期望順序執(zhí)行

背景

bean的生命周期如下圖所示:

image.png

@PostConstruct注解是會(huì)被一個(gè)專門的BeanPostProcessor接口的具體實(shí)現(xiàn)類來處理的掂咒,實(shí)現(xiàn)類是:InitDestroyAnnotationBeanPostProcessor侍匙。

按照加載順序,@PostConstruct也會(huì)按照依賴順序執(zhí)行觅捆。

但是在代碼里并沒有按照期望順序執(zhí)行赦役,依賴關(guān)系如下:

@Service
@Slf4j
public class A {
    @Resource
    private B b;

    @PostConstruct
    public void init(){
        log.info("A PostConstruct");
        b.test();
    }
}



@Service
@Slf4j
public class B {
    @Resource
    private C c;

    @PostConstruct
    public void init(){
        log.info("B PostConstruct");
    }

    public void test(){
        log.info("B Test");
    }
}


@Service
public class C {

    @Resource
    private A a;
}

A對(duì)象里依賴了B對(duì)象,并且A的@PostConstruct方法依賴了B的@PostConstruct生成的數(shù)據(jù)栅炒,但是A的@PostConstruct掂摔,導(dǎo)致拿到的數(shù)據(jù)為空

問題分析

經(jīng)過debug,發(fā)現(xiàn)是由于循環(huán)依賴導(dǎo)致职辅,B先加載并且提前暴露棒呛,導(dǎo)致A執(zhí)行@PostConstruct時(shí)B還沒有初始化完成

為解決循環(huán)依賴,spring采用三級(jí)緩存機(jī)制域携,將實(shí)例化成功的對(duì)象加載到三級(jí)緩存簇秒,

這三級(jí)緩存的作用分別是:

singletonFactories : 進(jìn)入實(shí)例化階段的單例對(duì)象工廠的cache (三級(jí)緩存)

earlySingletonObjects :完成實(shí)例化但是尚未初始化的,提前暴光的單例對(duì)象的Cache (二級(jí)緩存)

singletonObjects:完成初始化的單例對(duì)象的cache(一級(jí)緩存)

我們?cè)趧?chuàng)建bean的時(shí)候秀鞭,會(huì)首先從cache中獲取這個(gè)bean趋观,這個(gè)緩存就是sigletonObjects。主要的調(diào)用方法是:

protected Object getSingleton(String beanName, boolean allowEarlyReference) {
    Object singletonObject = this.singletonObjects.get(beanName);
    //isSingletonCurrentlyInCreation()判斷當(dāng)前單例bean是否正在創(chuàng)建中
    if (singletonObject == null && isSingletonCurrentlyInCreation(beanName)) {
        synchronized (this.singletonObjects) {
            singletonObject = this.earlySingletonObjects.get(beanName);
            //allowEarlyReference 是否允許從singletonFactories中通過getObject拿到對(duì)象
            if (singletonObject == null && allowEarlyReference) {
                ObjectFactory<?> singletonFactory = this.singletonFactories.get(beanName);
                if (singletonFactory != null) {
                    singletonObject = singletonFactory.getObject();
                    //其實(shí)也就是從三級(jí)緩存移動(dòng)到了二級(jí)緩存
                    this.earlySingletonObjects.put(beanName, singletonObject);
                    this.singletonFactories.remove(beanName);
                }
            }
        }
    }
    return (singletonObject != NULL_OBJECT ? singletonObject : null);
}

從上面三級(jí)緩存的分析锋边,我們可以知道皱坛,Spring解決循環(huán)依賴的訣竅就在于singletonFactories這個(gè)三級(jí)cache。這個(gè)cache的類型是ObjectFactory豆巨,定義如下:

public interface ObjectFactory<T> {
    T getObject() throws BeansException;
}

這個(gè)接口在AbstractBeanFactory里實(shí)現(xiàn)剩辟,并在核心方法doCreateBean()引用下面的方法:

protected void addSingletonFactory(String beanName, ObjectFactory<?> singletonFactory) {
    Assert.notNull(singletonFactory, "Singleton factory must not be null");
    synchronized (this.singletonObjects) {
        if (!this.singletonObjects.containsKey(beanName)) {
            this.singletonFactories.put(beanName, singletonFactory);
            this.earlySingletonObjects.remove(beanName);
            this.registeredSingletons.add(beanName);
        }
    }
}

這段代碼發(fā)生在createBeanInstance之后,populateBean()之前往扔,也就是說單例對(duì)象此時(shí)已經(jīng)被創(chuàng)建出來(調(diào)用了構(gòu)造器)贩猎。這個(gè)對(duì)象已經(jīng)被生產(chǎn)出來了,此時(shí)將這個(gè)對(duì)象提前曝光出來

所以在發(fā)生循環(huán)依賴時(shí)萍膛,B還未初始化吭服,所以@PostConstruct方法還未執(zhí)行

解決方案

  • 解決循環(huán)依賴

  • 將邏輯從@PostConstruct里抽出來

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個(gè)濱河市蝗罗,隨后出現(xiàn)的幾起案子艇棕,更是在濱河造成了極大的恐慌蝌戒,老刑警劉巖,帶你破解...
    沈念sama閱讀 216,402評(píng)論 6 499
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件沼琉,死亡現(xiàn)場(chǎng)離奇詭異北苟,居然都是意外死亡,警方通過查閱死者的電腦和手機(jī)刺桃,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 92,377評(píng)論 3 392
  • 文/潘曉璐 我一進(jìn)店門粹淋,熙熙樓的掌柜王于貴愁眉苦臉地迎上來吸祟,“玉大人瑟慈,你說我怎么就攤上這事∥葚埃” “怎么了葛碧?”我有些...
    開封第一講書人閱讀 162,483評(píng)論 0 353
  • 文/不壞的土叔 我叫張陵,是天一觀的道長过吻。 經(jīng)常有香客問我进泼,道長,這世上最難降的妖魔是什么纤虽? 我笑而不...
    開封第一講書人閱讀 58,165評(píng)論 1 292
  • 正文 為了忘掉前任乳绕,我火速辦了婚禮,結(jié)果婚禮上逼纸,老公的妹妹穿的比我還像新娘洋措。我一直安慰自己,他們只是感情好杰刽,可當(dāng)我...
    茶點(diǎn)故事閱讀 67,176評(píng)論 6 388
  • 文/花漫 我一把揭開白布菠发。 她就那樣靜靜地躺著,像睡著了一般贺嫂。 火紅的嫁衣襯著肌膚如雪滓鸠。 梳的紋絲不亂的頭發(fā)上,一...
    開封第一講書人閱讀 51,146評(píng)論 1 297
  • 那天第喳,我揣著相機(jī)與錄音糜俗,去河邊找鬼。 笑死曲饱,一個(gè)胖子當(dāng)著我的面吹牛悠抹,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播渔工,決...
    沈念sama閱讀 40,032評(píng)論 3 417
  • 文/蒼蘭香墨 我猛地睜開眼锌钮,長吁一口氣:“原來是場(chǎng)噩夢(mèng)啊……” “哼!你這毒婦竟也來了引矩?” 一聲冷哼從身側(cè)響起梁丘,我...
    開封第一講書人閱讀 38,896評(píng)論 0 274
  • 序言:老撾萬榮一對(duì)情侶失蹤侵浸,失蹤者是張志新(化名)和其女友劉穎,沒想到半個(gè)月后氛谜,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體掏觉,經(jīng)...
    沈念sama閱讀 45,311評(píng)論 1 310
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 37,536評(píng)論 2 332
  • 正文 我和宋清朗相戀三年值漫,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了澳腹。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點(diǎn)故事閱讀 39,696評(píng)論 1 348
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡杨何,死狀恐怖酱塔,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情危虱,我是刑警寧澤羊娃,帶...
    沈念sama閱讀 35,413評(píng)論 5 343
  • 正文 年R本政府宣布,位于F島的核電站埃跷,受9級(jí)特大地震影響蕊玷,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜弥雹,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,008評(píng)論 3 325
  • 文/蒙蒙 一垃帅、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧剪勿,春花似錦贸诚、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 31,659評(píng)論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至赴涵,卻和暖如春媒怯,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背髓窜。 一陣腳步聲響...
    開封第一講書人閱讀 32,815評(píng)論 1 269
  • 我被黑心中介騙來泰國打工扇苞, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留,地道東北人寄纵。 一個(gè)月前我還...
    沈念sama閱讀 47,698評(píng)論 2 368
  • 正文 我出身青樓鳖敷,卻偏偏與公主長得像,于是被迫代替她去往敵國和親程拭。 傳聞我的和親對(duì)象是個(gè)殘疾皇子定踱,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 44,592評(píng)論 2 353