Spring 系列(六) lazy-init延遲加載原理

PS: 查看源碼的jar版本為 Spring-beans:5.2.15.release

普通Bean的初始化是在容器啟動(dòng)初始化階段執(zhí)行的,而被lazy-init=true修飾的bean則是從容器第一次進(jìn)行context.getBean()進(jìn)行觸發(fā)。Spring啟動(dòng)的時(shí)候會(huì)把所有bean信息(包括XML和注解)解析轉(zhuǎn)化為Spring能夠識(shí)別的BeanDefinition并存到HashMap里供下面的初始化時(shí)用,然后會(huì)對(duì)每個(gè)BeanDefinition進(jìn)行從處理,如果時(shí)懶加載的則在容器初始化階段不處理,其他的則在容器初始化階段進(jìn)行初始化并依賴注入。

    @Override
    public void preInstantiateSingletons() throws BeansException {
        if (logger.isTraceEnabled()) {
            logger.trace("Pre-instantiating singletons in " + this);
        }

        //所有beanDefinition集合
        List<String> beanNames = new ArrayList<>(this.beanDefinitionNames);

        // 觸發(fā)所有非懶加載單例bean的初始化
        for (String beanName : beanNames) {
            //獲取bean定義
            RootBeanDefinition bd = getMergedLocalBeanDefinition(beanName);
            //判斷是否是懶加載單例bean,如果是單例的并且不是懶加載的則在容器創(chuàng)建時(shí)初始化
            if (!bd.isAbstract() && bd.isSingleton() && !bd.isLazyInit()) {
                //判斷是否 FactoryBean
                if (isFactoryBean(beanName)) {
                    Object bean = getBean(FACTORY_BEAN_PREFIX + beanName);
                    if (bean instanceof FactoryBean) {
                        FactoryBean<?> factory = (FactoryBean<?>) bean;
                        boolean isEagerInit;
                        if (System.getSecurityManager() != null && factory instanceof SmartFactoryBean) {
                            isEagerInit = AccessController.doPrivileged(
                                    (PrivilegedAction<Boolean>) ((SmartFactoryBean<?>) factory)::isEagerInit,
                                    getAccessControlContext());
                        }
                        else {
                            isEagerInit = (factory instanceof SmartFactoryBean &&
                                    ((SmartFactoryBean<?>) factory).isEagerInit());
                        }
                        if (isEagerInit) {
                            getBean(beanName);
                        }
                    }
                }
                else {
/*
如果普通bean則進(jìn)行初始化并依賴注入,getBean(beanName)接下來(lái)觸發(fā)的邏輯和懶加載時(shí) 
context.getBean("beanName") 所觸發(fā)的邏輯是?樣的
*/                  getBean(beanName);
                }
            }
        }
...
}
  • 總結(jié)
    • 對(duì)于被修飾為lazy-init的bean Spring容器初始化階段不會(huì)進(jìn)行init并且依賴注入,當(dāng)?shù)谝淮蝕etBean時(shí)候才會(huì)進(jìn)行初始化并依賴注入
    • 對(duì)于非懶加載的bean,getBean的時(shí)候會(huì)從緩存里獲取,因?yàn)槿萜鞒跏蓟A段Bean已經(jīng)初始胡完成并緩存起來(lái)了
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
【社區(qū)內(nèi)容提示】社區(qū)部分內(nèi)容疑似由AI輔助生成,瀏覽時(shí)請(qǐng)結(jié)合常識(shí)與多方信息審慎甄別。
平臺(tái)聲明:文章內(nèi)容(如有圖片或視頻亦包括在內(nèi))由作者上傳并發(fā)布,文章內(nèi)容僅代表作者本人觀點(diǎn),簡(jiǎn)書(shū)系信息發(fā)布平臺(tái),僅提供信息存儲(chǔ)服務(wù)。

相關(guān)閱讀更多精彩內(nèi)容

  • lazy-init 延遲加載機(jī)制分析 普通 Bean 的初始化是在容器啟動(dòng)初始化階段執(zhí)行的,而被lazy-init...
    雪礪青松閱讀 504評(píng)論 0 1
  • User.java未實(shí)例化示例 讀取bean配置的lazy-init標(biāo)簽 讀取lazy-init標(biāo)簽是伴隨著讀取b...
    sunpy閱讀 429評(píng)論 0 0
  • 五、IoC容器的高級(jí)特性 1、介紹 通過(guò)前面4篇文章對(duì)SpringIoC容器的源碼分析,我們已經(jīng)基本上了解了Spr...
    逐夢(mèng)余生閱讀 1,235評(píng)論 0 0
  • Spring的IoC能力是我們?cè)陧?xiàng)目中經(jīng)常用到的,這一塊的相關(guān)原理很有必要搞清楚。Spring的IoC功能實(shí)現(xiàn)可分...
    Justlearn閱讀 718評(píng)論 0 1
  • 如果對(duì)IOC的概念還不是很清楚,可以先閱讀上一篇文章:Spring_IOC_01——概念講解 Spring IOC...
    _Zy閱讀 1,236評(píng)論 3 12

友情鏈接更多精彩內(nèi)容