spring源碼解析八(創(chuàng)建單例bean)

上一節(jié),研究了下獲取bean的源碼,邏輯還是比較簡單的,這次,我們一起來研究下創(chuàng)建bean的源碼,話不多說,我們直接看源碼
發(fā)現(xiàn)bean的創(chuàng)建的代碼都是在AbstractAutowireCapableBeanFactory這個類中,之前分析@Autowired注解的時候,也是在這個類中
createBean的其他方法我在這里就不分析了,咱們具體就分析doCreateBean

AbstractAutowireCapableBeanFactory

這里將成三個步驟,創(chuàng)建bean實例,填充bean的屬性,執(zhí)行后置處理器,
···
@Override
protected Object createBean(String beanName, RootBeanDefinition mbd, @Nullable Object[] args)
throws BeanCreationException {

    RootBeanDefinition mbdToUse = mbd;

    //確保當(dāng)前的bean類已經(jīng)被解析
    Class<?> resolvedClass = resolveBeanClass(mbd, beanName);
    if (resolvedClass != null && !mbd.hasBeanClass() && mbd.getBeanClassName() != null) {
        mbdToUse = new RootBeanDefinition(mbd);
        mbdToUse.setBeanClass(resolvedClass);
    }
        mbdToUse.prepareMethodOverrides();

    try {
        //1:在獲取bean之前獲取一下嘗試獲取下代理對象
        //2:判斷是不是基本類型的對象
        //3:是否需要跳過(獲取增強對象,判斷存不存在@Aspect注解,存在,證明是AOP類,判斷切點,根據(jù)@Before等注解獲取一個完整的增強對象)
        //4:將獲取到的增強對象集合與緩存緩存中的advisor.getName執(zhí)行比較,如果是同一個bean,直接返回
        //判斷是否有自定義的targetSource,存在的話,targetSource以自定義的方式處理目標(biāo)類,然后返回,否則返回null
        Object bean = resolveBeforeInstantiation(beanName, mbdToUse);
        if (bean != null) {
            return bean;
        }
    }

    try {
        //真正創(chuàng)建bean實例的方法,
        Object beanInstance = doCreateBean(beanName, mbdToUse, args);
        return beanInstance;
    }

}

···
這個方法很長,咱們先簡單看一下這個方法到底做了什么操作,為了看得清楚,我會刪除部分拋異常代碼
1:從緩存中獲取bean的包裝對象并刪除緩存
2:緩存中獲取不到bean,創(chuàng)建bean實例
3:解決循環(huán)依賴
4:填充bean中的屬性
5:重新注冊一次bean實例
···

AbstractAutowireCapableBeanFactory

doCreateBean

protected Object doCreateBean(final String beanName, final RootBeanDefinition mbd, final @Nullable Object[] args)
throws BeanCreationException {

    // 1:創(chuàng)建bean包裝對象,
    BeanWrapper instanceWrapper = null;
    if (mbd.isSingleton()) {
        //如果是單例,從緩存中獲取bean包裝對象,并刪除緩存
        instanceWrapper = this.factoryBeanInstanceCache.remove(beanName);
    }
    if (instanceWrapper == null) {
        //創(chuàng)建bean實例,并返回bean包裝對象
        //這里創(chuàng)建bean實例并返回bean包裝對象
        //createBeanInstance方法中會通過三種方式創(chuàng)建bean實例
        //1:通過工廠方法創(chuàng)建
        //2:通過構(gòu)造方法注入,比如autowire或者constructor的方式創(chuàng)建bean實例
        //3:通過無參構(gòu)造的方式創(chuàng)建bean實例
        //4:如果配置文件中配置了lookup-method 和 replace-method,則會利用Cglib代理的方式增強bean實例
        instanceWrapper = createBeanInstance(beanName, mbd, args);
    }
    final Object bean = instanceWrapper.getWrappedInstance();
    Class<?> beanType = instanceWrapper.getWrappedClass();
    if (beanType != NullBean.class) {
        mbd.resolvedTargetType = beanType;
    }

    // 后置處理器處理流程,就不展開說了
    synchronized (mbd.postProcessingLock) {
        if (!mbd.postProcessed) {
            try {
                applyMergedBeanDefinitionPostProcessors(mbd, beanType, beanName);
            }
            mbd.postProcessed = true;
        }
    }
    //===========================================================================================
    //2.1:判斷是否存在循環(huán)引用,
    //這個earlySingletonExposure的意思是:單例bean是否提前暴露
    //同時滿足三個條件,才會變?yōu)閠rue
    //1:mbd.isSingleton():是否是單例
    //2:this.allowCircularReferences:是否循環(huán)引用
    //3:isSingletonCurrentlyInCreation:該單例bean是否處于創(chuàng)建中
    boolean earlySingletonExposure = (mbd.isSingleton() && this.allowCircularReferences &&
            isSingletonCurrentlyInCreation(beanName));
    if (earlySingletonExposure) {
        //添加單例工廠對象,用于處理循環(huán)引用,這個地方是利用了函數(shù)式接口,我們把它展開看,就好看一點
        addSingletonFactory(beanName, () -> getEarlyBeanReference(beanName, mbd, bean));

        addSingletonFactory(beanName, new ObjectFactory<Object>() {
                    @Override
                    public Object getObject() throws BeansException {
                        // 獲取早期bean引用
                        return getEarlyBeanReference(beanName, mbd, bean);
                    }
                });
    }

    //=================================================================================================
    //3:填充屬性
    Object exposedObject = bean;
        populateBean(beanName, mbd, instanceWrapper);
        exposedObject = initializeBean(beanName, exposedObject, mbd);

    //2.2解決循環(huán)引用
    if (earlySingletonExposure) {
        Object earlySingletonReference = getSingleton(beanName, false);
        if (earlySingletonReference != null) {
            if (exposedObject == bean) {
                exposedObject = earlySingletonReference;
            }
            else if (!this.allowRawInjectionDespiteWrapping && hasDependentBean(beanName)) {
                String[] dependentBeans = getDependentBeans(beanName);
                Set<String> actualDependentBeans = new LinkedHashSet<>(dependentBeans.length);
                for (String dependentBean : dependentBeans) {
                    if (!removeSingletonIfCreatedForTypeCheckOnly(dependentBean)) {
                        actualDependentBeans.add(dependentBean);
                    }
                }
                if (!actualDependentBeans.isEmpty()) {
                    throw new BeanCurrentlyInCreationException(beanName,
                            "Bean with name '" + beanName + "' has been injected into other beans [" +
                            StringUtils.collectionToCommaDelimitedString(actualDependentBeans) +
                            "] in its raw version as part of a circular reference, but has eventually been " +
                            "wrapped. This means that said other beans do not use the final version of the " +
                            "bean. This is often the result of over-eager type matching - consider using " +
                            "'getBeanNamesOfType' with the 'allowEagerInit' flag turned off, for example.");
                }
            }
        }
    }

    // 必要時再注冊一次bean
    registerDisposableBeanIfNecessary(beanName, bean, mbd);

    return exposedObject;
}

···

?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末员魏,一起剝皮案震驚了整個濱河市,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌,老刑警劉巖,帶你破解...
    沈念sama閱讀 222,729評論 6 517
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場離奇詭異,居然都是意外死亡用押,警方通過查閱死者的電腦和手機(jī),發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 95,226評論 3 399
  • 文/潘曉璐 我一進(jìn)店門靶剑,熙熙樓的掌柜王于貴愁眉苦臉地迎上來只恨,“玉大人,你說我怎么就攤上這事抬虽」倜伲” “怎么了?”我有些...
    開封第一講書人閱讀 169,461評論 0 362
  • 文/不壞的土叔 我叫張陵阐污,是天一觀的道長休涤。 經(jīng)常有香客問我,道長笛辟,這世上最難降的妖魔是什么功氨? 我笑而不...
    開封第一講書人閱讀 60,135評論 1 300
  • 正文 為了忘掉前任,我火速辦了婚禮手幢,結(jié)果婚禮上捷凄,老公的妹妹穿的比我還像新娘。我一直安慰自己围来,他們只是感情好跺涤,可當(dāng)我...
    茶點故事閱讀 69,130評論 6 398
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著监透,像睡著了一般桶错。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上胀蛮,一...
    開封第一講書人閱讀 52,736評論 1 312
  • 那天院刁,我揣著相機(jī)與錄音,去河邊找鬼粪狼。 笑死退腥,一個胖子當(dāng)著我的面吹牛任岸,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播狡刘,決...
    沈念sama閱讀 41,179評論 3 422
  • 文/蒼蘭香墨 我猛地睜開眼享潜,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了颓帝?” 一聲冷哼從身側(cè)響起,我...
    開封第一講書人閱讀 40,124評論 0 277
  • 序言:老撾萬榮一對情侶失蹤窝革,失蹤者是張志新(化名)和其女友劉穎购城,沒想到半個月后,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體虐译,經(jīng)...
    沈念sama閱讀 46,657評論 1 320
  • 正文 獨居荒郊野嶺守林人離奇死亡瘪板,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 38,723評論 3 342
  • 正文 我和宋清朗相戀三年,在試婚紗的時候發(fā)現(xiàn)自己被綠了漆诽。 大學(xué)時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片侮攀。...
    茶點故事閱讀 40,872評論 1 353
  • 序言:一個原本活蹦亂跳的男人離奇死亡,死狀恐怖厢拭,靈堂內(nèi)的尸體忽然破棺而出兰英,到底是詐尸還是另有隱情,我是刑警寧澤供鸠,帶...
    沈念sama閱讀 36,533評論 5 351
  • 正文 年R本政府宣布畦贸,位于F島的核電站,受9級特大地震影響楞捂,放射性物質(zhì)發(fā)生泄漏薄坏。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點故事閱讀 42,213評論 3 336
  • 文/蒙蒙 一寨闹、第九天 我趴在偏房一處隱蔽的房頂上張望胶坠。 院中可真熱鬧,春花似錦繁堡、人聲如沸沈善。這莊子的主人今日做“春日...
    開封第一講書人閱讀 32,700評論 0 25
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽矮瘟。三九已至,卻和暖如春塑娇,著一層夾襖步出監(jiān)牢的瞬間澈侠,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 33,819評論 1 274
  • 我被黑心中介騙來泰國打工埋酬, 沒想到剛下飛機(jī)就差點兒被人妖公主榨干…… 1. 我叫王不留哨啃,地道東北人烧栋。 一個月前我還...
    沈念sama閱讀 49,304評論 3 379
  • 正文 我出身青樓,卻偏偏與公主長得像拳球,于是被迫代替她去往敵國和親审姓。 傳聞我的和親對象是個殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點故事閱讀 45,876評論 2 361