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

Spring中的InitializingBean接口源碼解析

 更新時(shí)間:2024年02月14日 09:09:13   作者:securitit  
這篇文章主要介紹了Spring中的InitializingBean接口源碼解析,InitializingBean接口為Bean初始化提供了一種方式,實(shí)現(xiàn)InitializingBean接口的Bean,在BeanFactory設(shè)置其所有屬性后會(huì)調(diào)用其afterPropertiesSet()方法,需要的朋友可以參考下

簡(jiǎn)介

InitializingBean

InitializingBean接口為Bean初始化提供了一種方式。

實(shí)現(xiàn)InitializingBean接口的Bean,在BeanFactory設(shè)置其所有屬性后會(huì)調(diào)用其afterPropertiesSet()方法??梢栽赼fterPropertiesSet()方法中執(zhí)行自定義初始化、屬性檢查或強(qiáng)制校驗(yàn)等,若不滿足要求可以拋出異常以中斷Spring的加載流程。

InitializingBean應(yīng)用時(shí)有幾點(diǎn)需要注意:

① Bean必須實(shí)現(xiàn)InitializingBean接口。

② Bean的afterPropertiesSet不能使用@PostConstruct注釋。

init-method

init-method定義初始化方法為Bean初始化提供了另一種方式。

Bean聲明時(shí)配置init-method屬性,用于指定初始化方法。與InitializingBean方式類似,在BeanFactory設(shè)置其所有屬性后會(huì)調(diào)用其init-method指定的方法。可以在init-method方法中執(zhí)行自定義初始化、屬性檢查或強(qiáng)制校驗(yàn)等,若不滿足要求可以拋出異常以中斷Spring的加載流程。

init-method應(yīng)用時(shí)有幾個(gè)限制需要注意:

① init-method指定屬性不能為空。

② Bean不可以實(shí)現(xiàn)InitializingBean接口或Bean的init-method方法名不可以為afterPropertiesSet。

③ Bean的init-method方法不能使用@PostConstruct注釋。

演示示例

InitializingBean和init-method可以作用于同一個(gè)Bean,但是需要滿足上面所羅列的注意事項(xiàng),下面來(lái)使用一個(gè)簡(jiǎn)單示例看一下。

1) 建Bean,實(shí)現(xiàn)InitializingBean接口,并額外填加一個(gè)方法用于init-method配置。

package com.arhorchin.securitit.initbean;

import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.springframework.beans.factory.InitializingBean;

/**
 * @author Securitit.
 * @note Bean初始化測(cè)試.
 */
public class InitTestBean implements InitializingBean {

    /**
     * logger.
     */
    private Logger logger = LoggerFactory.getLogger(InitTestBean.class);
    
    @Override
    public void afterPropertiesSet() throws Exception {
        logger.info("調(diào)用InitializingBean的afterPropertiesSet方法.");
    }
    
    public void initMethod() throws Exception {
        logger.info("調(diào)用init-method的initMethod方法.");
    }

}

2) 在Spring的配置文件中增加Bean聲明,并指定init-method屬性。

<bean class="com.arhorchin.securitit.initbean.InitTestBean" init-method="initMethod"></bean>

3) 運(yùn)行程序查看效果,可以看到如下的輸出。

2020-12-09 10:58:29 INFO [c.a.s.i.InitTestBean] 調(diào)用InitializingBean的afterPropertiesSet方法.
2020-12-09 10:58:29 INFO [c.a.s.i.InitTestBean] 調(diào)用init-method的initMethod方法.

從結(jié)果可以看到,InitializingBean的afterPropertiesSet先于Bean的init-method指定的方法調(diào)用。

源碼解析

InitializingBean和init-method源碼基本集中在org.springframework.beans.factory.support.AbstractAutowireCapableBeanFactory類中。

1)initializeBean(...)方法

AbstractAutowireCapableBeanFactory的initializeBean(...)方法

initializeBean(...)方法中針對(duì)Bean進(jìn)行了幾個(gè)操作:

① 若Bean實(shí)現(xiàn)了Aware接口,則觸發(fā)方法調(diào)用。包括:BeanNameAware、BeanClassLoaderAware和BeanFactoryAware。

② 調(diào)用注冊(cè)的BeanPostProcessor的postProcessBeforeInitialization(...)方法。

③ 調(diào)用初始化方法,包括InitializingBean的afterPropertiesSet()方法和Bean的init-method指定的方法。

④ 調(diào)用注冊(cè)的BeanPostProcessor的postProcessAfterInitialization(...)方法。

/**
 * 初始化給定的Bean實(shí)例,應(yīng)用工廠回調(diào)、init方法和Bean后處理程序.
 * 對(duì)于傳統(tǒng)定義的Bean,從createBean調(diào)用,對(duì)于現(xiàn)有的Bean實(shí)例從initializeBean調(diào)用.
 * @param beanName 工廠中的Bean名稱(用于調(diào)試).
 * @param bean 需要初始化新的Bean實(shí)例.
 * @param mbd 創(chuàng)建Bean時(shí)使用的Bean定義(如果給定現(xiàn)有的Bean實(shí)例,也可以是null).
 * @return 初始化的Bean實(shí)例(可能被包裝).
 */
protected Object initializeBean(final String beanName, final Object bean, @Nullable RootBeanDefinition mbd) {
    // 若Bean實(shí)現(xiàn)了Aware接口,則觸發(fā)方法調(diào)用.
    if (System.getSecurityManager() != null) {
        AccessController.doPrivileged((PrivilegedAction<Object>) () -> {
            invokeAwareMethods(beanName, bean);
            return null;
        }, getAccessControlContext());
    }
    else {
        invokeAwareMethods(beanName, bean);
    }

    Object wrappedBean = bean;
    // 在Bean初始化前處理BeanPostProcessor.
    if (mbd == null || !mbd.isSynthetic()) {
        wrappedBean = applyBeanPostProcessorsBeforeInitialization(wrappedBean, beanName);
    }
    // 進(jìn)行Bean初始化,包括如下兩種方式:
    // 1.調(diào)用InitializingBean.afterPropertiesSet()方法.
    // 2.調(diào)用Bean配置的init-method方法.
    try {
        invokeInitMethods(beanName, wrappedBean, mbd);
    }
    catch (Throwable ex) {
        throw new BeanCreationException(
            (mbd != null ? mbd.getResourceDescription() : null),
            beanName, "Invocation of init method failed", ex);
    }
    // 在Bean初始化后處理BeanPostProcessor.
    if (mbd == null || !mbd.isSynthetic()) {
        wrappedBean = applyBeanPostProcessorsAfterInitialization(wrappedBean, beanName);
    }

    return wrappedBean;
}

2)實(shí)現(xiàn)Aware接口

若Bean實(shí)現(xiàn)了Aware接口,則觸發(fā)方法調(diào)用。包括:BeanNameAware、BeanClassLoaderAware和BeanFactoryAware

/**
 * 若Bean實(shí)現(xiàn)了Aware接口,則觸發(fā)方法調(diào)用.
 */
private void invokeAwareMethods(final String beanName, final Object bean) {
    if (bean instanceof Aware) {
        // 調(diào)用BeanNameAware.setBeanName(...)方法.
        if (bean instanceof BeanNameAware) {
            ((BeanNameAware) bean).setBeanName(beanName);
        }
        // 調(diào)用BeanClassLoaderAware.setBeanClassLoader(...)方法.
        if (bean instanceof BeanClassLoaderAware) {
            ClassLoader bcl = getBeanClassLoader();
            if (bcl != null) {
                ((BeanClassLoaderAware) bean).setBeanClassLoader(bcl);
            }
        }
        // 調(diào)用BeanFactoryAware.setBeanFactory(...).
        if (bean instanceof BeanFactoryAware) {
            ((BeanFactoryAware) bean).setBeanFactory(AbstractAutowireCapableBeanFactory.this);
        }
    }
}

3)InitializingBean的afterPropertiesSet方法調(diào)用

invokeInitMethods主要用于InitializingBean的afterPropertiesSet方法調(diào)用,從方法源碼中也可以看到,是先調(diào)用InitializingBean的afterPropertiesSet方法,然后再調(diào)用Bean的init-method指定的方法,查看代碼的注釋,可以看到相關(guān)的內(nèi)容,不做過(guò)多解析。

/**
 * 所有屬性設(shè)置完成后可選的Bean初始化.
 * Bean實(shí)現(xiàn)了InitializingBean接口或定義了init-method方法,則會(huì)進(jìn)行回調(diào)處理.
 * @param beanName 工廠中的Bean名稱(用于調(diào)試).
 * @param bean 需要初始化新的Bean實(shí)例.
 * @param mbd 創(chuàng)建Bean時(shí)使用的合并Bean定義(如果給定現(xiàn)有的Bean實(shí)例,也可以是null).
 */
protected void invokeInitMethods(String beanName, final Object bean, @Nullable RootBeanDefinition mbd)
    throws Throwable {
    // Bean是否實(shí)現(xiàn)了InitializingBean接口.
    boolean isInitializingBean = (bean instanceof InitializingBean);
    // 調(diào)用InitializingBean.afterPropertiesSet()方法.需要滿足條件:
    // 1.Bean實(shí)現(xiàn)了InitializingBean接口.
    // 2.Bean為空或afterPropertiesSet方法未被@PostConstruct注釋.
    if (isInitializingBean && (mbd == null || !mbd.isExternallyManagedInitMethod("afterPropertiesSet"))) {
        if (logger.isDebugEnabled()) {
            logger.debug("Invoking afterPropertiesSet() on bean with name '" + beanName + "'");
        }
        // 調(diào)用InitializingBean.afterPropertiesSet()方法.
        if (System.getSecurityManager() != null) {
            try {
                AccessController.doPrivileged((PrivilegedExceptionAction<Object>) () -> {
                    ((InitializingBean) bean).afterPropertiesSet();
                    return null;
                }, getAccessControlContext());
            }
            catch (PrivilegedActionException pae) {
                throw pae.getException();
            }
        }
        else {
            ((InitializingBean) bean).afterPropertiesSet();
        }
    }
    // 調(diào)用配置init-method方法處理.
    if (mbd != null && bean.getClass() != NullBean.class) {
        String initMethodName = mbd.getInitMethodName();
        // 1.init-method配置不能為空.
        // 2.Bean不能實(shí)現(xiàn)InitializingBean或init-method方法不是afterPropertiesSet.
        // 3.init-method方法未被@PostConstruct注釋.
        if (StringUtils.hasLength(initMethodName) &&
            !(isInitializingBean && "afterPropertiesSet".equals(initMethodName)) &&
            !mbd.isExternallyManagedInitMethod(initMethodName)) {
            invokeCustomInitMethod(beanName, bean, mbd);
        }
    }
}

4)invokeCustomInitMethod

invokeCustomInitMethod主要用于調(diào)用init-method指定的方法,調(diào)用方式僅是通過(guò)反射來(lái)調(diào)用,查看代碼的注釋,可以看到相關(guān)的內(nèi)容,不做過(guò)多解析。

/**
 * 在給定的Bean上調(diào)用指定的自定義init方法.由invokeInitMethods調(diào)用.
 * 可以在子類中重寫(xiě),以便使用參數(shù)自定義解析init方法.
 */
protected void invokeCustomInitMethod(String beanName, final Object bean, RootBeanDefinition mbd)
    throws Throwable {
    // 取得Method實(shí)例.
    String initMethodName = mbd.getInitMethodName();
    Assert.state(initMethodName != null, "No init method set");
    final Method initMethod = (mbd.isNonPublicAccessAllowed() ?
                               BeanUtils.findMethod(bean.getClass(), initMethodName) :
                               ClassUtils.getMethodIfAvailable(bean.getClass(), initMethodName));
    if (initMethod == null) {
        if (mbd.isEnforceInitMethod()) {
            throw new BeanDefinitionValidationException("Couldn't find an init method named '" +
                                                        initMethodName + "' on bean with name '" + beanName + "'");
        }
        else {
            if (logger.isDebugEnabled()) {
                logger.debug("No default init method named '" + initMethodName +
                             "' found on bean with name '" + beanName + "'");
            }
            // Ignore non-existent default lifecycle methods.
            return;
        }
    }
    if (logger.isDebugEnabled()) {
        logger.debug("Invoking init method  '" + initMethodName + "' on bean with name '" + beanName + "'");
    }
    // 調(diào)用Bean的init-method配置的方法.
    if (System.getSecurityManager() != null) {
        AccessController.doPrivileged((PrivilegedAction<Object>) () -> {
            ReflectionUtils.makeAccessible(initMethod);
            return null;
        });
        try {
            AccessController.doPrivileged((PrivilegedExceptionAction<Object>) () ->
                                          initMethod.invoke(bean), getAccessControlContext());
        }
        catch (PrivilegedActionException pae) {
            InvocationTargetException ex = (InvocationTargetException) pae.getException();
            throw ex.getTargetException();
        }
    }
    else {
        try {
            ReflectionUtils.makeAccessible(initMethod);
            initMethod.invoke(bean);
        }
        catch (InvocationTargetException ex) {
            throw ex.getTargetException();
        }
    }
}

總結(jié)

InitializingBean是一個(gè)很神奇的接口,Spring框架中對(duì)InitializingBean的應(yīng)用很是頻繁,init-method同樣如此,一定要了解兩者之間的調(diào)用順序,才能在更細(xì)粒度控制Bean的初始化過(guò)程。

源碼解析基于spring-framework-5.0.5.RELEASE版本源碼。

若文中存在錯(cuò)誤和不足,歡迎指正!

到此這篇關(guān)于Spring中的InitializingBean接口源碼解析的文章就介紹到這了,更多相關(guān)InitializingBean接口 內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!

相關(guān)文章

  • mybatis分頁(yè)效果實(shí)現(xiàn)代碼

    mybatis分頁(yè)效果實(shí)現(xiàn)代碼

    這篇文章主要為大家詳細(xì)介紹了mybatis分頁(yè)效果的實(shí)現(xiàn)代碼,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下
    2017-04-04
  • Maven私服倉(cāng)庫(kù)Nexus配置小結(jié)

    Maven私服倉(cāng)庫(kù)Nexus配置小結(jié)

    Maven 私服是一種特殊的Maven遠(yuǎn)程倉(cāng)庫(kù),它是架設(shè)在局域網(wǎng)內(nèi)的倉(cāng)庫(kù)服務(wù),本文就來(lái)介紹一下Maven私服倉(cāng)庫(kù)Nexus配置小結(jié),具有一定的參考價(jià)值,感興趣的可以了解一下
    2024-08-08
  • springboot啟動(dòng)過(guò)程中常用的回調(diào)示例詳解

    springboot啟動(dòng)過(guò)程中常用的回調(diào)示例詳解

    springboot提供非常豐富回調(diào)接口,利用這些接口可以做非常多的事情,本文通過(guò)實(shí)例代碼給大家介紹springboot啟動(dòng)過(guò)程中常用的回調(diào)知識(shí)感興趣的朋友跟隨小編一起看看吧
    2022-01-01
  • 基于hashmap 的擴(kuò)容和樹(shù)形化全面分析

    基于hashmap 的擴(kuò)容和樹(shù)形化全面分析

    這篇文章主要介紹了hashmap 的擴(kuò)容和樹(shù)形化的使用,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教
    2021-06-06
  • redis防止重復(fù)提交的實(shí)現(xiàn)示例

    redis防止重復(fù)提交的實(shí)現(xiàn)示例

    在開(kāi)發(fā)中我們都需要處理重復(fù)提交的問(wèn)題,本文主要介紹了redis防止重復(fù)提交的實(shí)現(xiàn)示例,具有一定的參考價(jià)值,感興趣的可以了解一下
    2024-06-06
  • Java確保MQ消息隊(duì)列不丟失的實(shí)現(xiàn)與流程分析

    Java確保MQ消息隊(duì)列不丟失的實(shí)現(xiàn)與流程分析

    在分布式系統(tǒng)中,消息隊(duì)列是核心組件之一,本文將探討如何確保MQ消息隊(duì)列不丟失,并通過(guò)Java代碼示例和流程圖來(lái)演示解決方案,需要的可以了解下
    2025-05-05
  • vscode 配置java環(huán)境并調(diào)試運(yùn)行的詳細(xì)過(guò)程

    vscode 配置java環(huán)境并調(diào)試運(yùn)行的詳細(xì)過(guò)程

    這篇文章主要介紹了vscode 配置java環(huán)境并調(diào)試運(yùn)行的詳細(xì)過(guò)程,本文給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下
    2021-05-05
  • 基于SpringBoot框架實(shí)現(xiàn)文件上傳下載分享功能

    基于SpringBoot框架實(shí)現(xiàn)文件上傳下載分享功能

    在當(dāng)今的Web應(yīng)用開(kāi)發(fā)中,文件上傳與下載功能是極為常見(jiàn)且重要的需求,無(wú)論是用戶上傳頭像、分享文檔,還是系統(tǒng)生成報(bào)告供用戶下載,都離不開(kāi)這一功能模塊,SpringBoot作為一款流行的Java開(kāi)發(fā)框架,為我們提供了簡(jiǎn)潔高效的方式來(lái)實(shí)現(xiàn)文件上傳與下載,需要的朋友可以參考下
    2025-06-06
  • 關(guān)于@SpringBootApplication詳解

    關(guān)于@SpringBootApplication詳解

    這篇文章主要介紹了關(guān)于@SpringBootApplication的使用方式,具有很好的參考價(jià)值,希望對(duì)大家有所幫助,如有錯(cuò)誤或未考慮完全的地方,望不吝賜教
    2024-08-08
  • SpringBoot實(shí)現(xiàn)緩存預(yù)熱的幾種常用方案

    SpringBoot實(shí)現(xiàn)緩存預(yù)熱的幾種常用方案

    緩存預(yù)熱是指在 Spring Boot 項(xiàng)目啟動(dòng)時(shí),預(yù)先將數(shù)據(jù)加載到緩存系統(tǒng)(如 Redis)中的一種機(jī)制,本文給大家介紹了SpringBoot實(shí)現(xiàn)緩存預(yù)熱的幾種常用方案,并通過(guò)代碼示例講解的非常詳細(xì),需要的朋友可以參考下
    2024-02-02

最新評(píng)論