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

SpringBoot ApplicationContextAware拓展接口使用詳解

 更新時(shí)間:2023年04月03日 09:56:27   作者:ForestSpringH  
當(dāng)一個(gè)類實(shí)現(xiàn)了這個(gè)接口(ApplicationContextAware)之后,這個(gè)類就可以方便獲得ApplicationContext中的所有bean。換句話說,就是這個(gè)類可以直接獲取spring配置文件中,所有有引用到的bean對(duì)象

近日沉醉于熟悉公司新項(xiàng)目的過程,不斷地接觸新的應(yīng)用場(chǎng)景與實(shí)現(xiàn)技術(shù),對(duì)于我是一種學(xué)不來的進(jìn)步,實(shí)踐是檢驗(yàn)真理的唯一標(biāo)準(zhǔn)。我們今天就淺淺的談一談springboot提供的16個(gè)拓展接口之一的ApplicationContextAware接口的應(yīng)用場(chǎng)景與實(shí)際作用!

ApplicationContextAware接口:

public interface ApplicationContextAware extends Aware {
    void setApplicationContext(ApplicationContext applicationContext) throws BeansException;
}

首先Aware接口就知道這是springboot擴(kuò)展給用戶使用的,這里提供了方法setApplicationContext,參數(shù)就是傳遞spring容器上下文對(duì)象進(jìn)來,我們可以接收這個(gè)上下文對(duì)象,我們要想知道獲取spring容器上下文ApplicationContext具體有什么作用,這才是擴(kuò)展接口的目的所在,獲取上下文根據(jù)上下文的特性做一些事情。

我們來看ApplicationContext對(duì)象的方法:

來看看AbstractApplicationContext實(shí)現(xiàn)類的方法:

    public Object getBean(String name) throws BeansException {this.assertBeanFactoryActive();return this.getBeanFactory().getBean(name);}
    public <T> T getBean(String name, Class<T> requiredType) throws BeansException {this.assertBeanFactoryActive();return this.getBeanFactory().getBean(name, requiredType);}
    public Object getBean(String name, Object... args) throws BeansException {this.assertBeanFactoryActive();return this.getBeanFactory().getBean(name, args);}
    public <T> T getBean(Class<T> requiredType) throws BeansException {this.assertBeanFactoryActive();return this.getBeanFactory().getBean(requiredType);}
    public <T> T getBean(Class<T> requiredType, Object... args) throws BeansException {this.assertBeanFactoryActive();return this.getBeanFactory().getBean(requiredType, args);}
    public <T> ObjectProvider<T> getBeanProvider(Class<T> requiredType) {this.assertBeanFactoryActive();return this.getBeanFactory().getBeanProvider(requiredType);}
    public <T> ObjectProvider<T> getBeanProvider(ResolvableType requiredType) {this.assertBeanFactoryActive();return this.getBeanFactory().getBeanProvider(requiredType);}
    public boolean containsBean(String name) {return this.getBeanFactory().containsBean(name);}
    public boolean isSingleton(String name) throws NoSuchBeanDefinitionException {this.assertBeanFactoryActive();return this.getBeanFactory().isSingleton(name);}
    public boolean isPrototype(String name) throws NoSuchBeanDefinitionException {this.assertBeanFactoryActive();return this.getBeanFactory().isPrototype(name);}

這里我們可以發(fā)現(xiàn) getBean()方法很眼熟,因?yàn)樵谧钭铋_始學(xué)習(xí)spring時(shí)沒有用spring的腳手架創(chuàng)建項(xiàng)目,我們獲取bean的方法通常是classPathContextLoader掃描bean的xml文件解析組成ApplicationCOntext對(duì)象,再調(diào)用它的getBean方法獲取實(shí)例bean。

由此可以發(fā)現(xiàn)我們主要的應(yīng)用途徑就是使用這個(gè)getBean的方法,那么動(dòng)態(tài)的注入bean我們通過很多方法就能實(shí)現(xiàn),所以這里不難想到,靜態(tài)方法中無法使用注入的bean的問題。

其次我們來復(fù)現(xiàn)這個(gè)問題,大家來看如下的代碼:

public class JsonGetter {
@Resource
private UuidGetter uuidGetter;
public static string Test(){
       return uuidGetter.getUuid();
}
public static JsONobject set0bjectToJsonObject(object data){
       return JsoNobject.parseObject(String.valueof(JsONObject.toJSON(data)));
}
public static JsONObject setStringTO3son0bject(String data) { return JsONObject.parseObject(data);
}

我們發(fā)現(xiàn)在靜態(tài)的Test方法中調(diào)用注入的bean直接報(bào)錯(cuò),這里解釋一下:歸功于類的加載機(jī)制與加載順序,靜態(tài)屬性與靜態(tài)代碼塊最先加載(static靜態(tài)優(yōu)先),這里加載靜態(tài)方法是沒有bean實(shí)例給你用的,自然會(huì)報(bào)錯(cuò)。

如何解決?我們可以采取Spring獲取bean對(duì)象時(shí)調(diào)用getBean方法的思路,在容器加載時(shí)將spring容器的上下文進(jìn)行靜態(tài)存儲(chǔ):

@Component
@Lazy(value = false)
public class SpringContextHolder implements ApplicationContextAware, DisposableBean {
    /**
     * 將上下文靜態(tài)設(shè)置,在初始化組件時(shí)就進(jìn)行靜態(tài)上下文的覆蓋(這個(gè)覆蓋是將遠(yuǎn)spring容器的上下文對(duì)象引用加到我們預(yù)定設(shè)置)
     */
    private static ApplicationContext applicationContext = null;
    public static ApplicationContext getApplicationContext() {
        assertContextInjected();
        return applicationContext;
    }
    @SuppressWarnings("unchecked")
    public static <T> T getBean(String name) {
        assertContextInjected();
        return (T) applicationContext.getBean(name);
    }
    public static  <T> T getBean(Class<T> beanType) {
        assertContextInjected();
        return applicationContext.getBean(beanType);
    }
    @Override
    public void setApplicationContext(@NotNull ApplicationContext applicationContext) throws BeansException {
        SpringContextHolder.applicationContext = applicationContext;
    }
    @Override
    public void destroy() {
        applicationContext = null;
    }
    private static void assertContextInjected() {
        Assert.notNull(applicationContext,
                "applicationContext屬性未注入, 請(qǐng)?jiān)赼pplicationContext.xml中定義SpringContextHolder.");
    }
    public static void pushEvent(ApplicationEvent event){
        assertContextInjected();
        applicationContext.publishEvent(event);
    }
}

這里只需要關(guān)注的是靜態(tài)成員變量ApplicationContext的定義、賦值與驗(yàn)證:

    /**
     * 將上下文靜態(tài)設(shè)置,在初始化組件時(shí)就進(jìn)行靜態(tài)上下文的覆蓋(這個(gè)覆蓋是將遠(yuǎn)spring容器的上下文對(duì)象引用加到我們預(yù)定設(shè)置)
     */
    private static ApplicationContext applicationContext = null;

重寫擴(kuò)展接口的方法,實(shí)現(xiàn)靜態(tài)上下文的覆蓋:

    @Override
    public void setApplicationContext(@NotNull ApplicationContext applicationContext) throws BeansException {
        SpringContextHolder.applicationContext = applicationContext;
    }

將獲取它的方法公有修飾,便于共享:

    public static ApplicationContext getApplicationContext() {
        assertContextInjected();
        return applicationContext;
    }

寫到這里還是不明白,這么定義一個(gè)組件,將spring上下文對(duì)象靜態(tài)覆蓋到底有何作用?

不要慌,我們來看看這個(gè)類的這個(gè)方法:

public class AppContext {
    static transient ThreadLocal<Map<String, String>> contextMap = new ThreadLocal<>();
    ......省略n行業(yè)務(wù)代碼
    public static void fillLoginContext() {
        DingAppInfo appInfo = SpringContextHolder.getBean(DingAppInfoService.class).findAppInfo(APP_CODE);
        setDingVerifyInfo(appInfo);
        CloudChatAppInfo cloudChatAppInfo = SpringContextHolder.getBean(CloudChatAppInfoService.class).findAppInfo(APP_CODE);
        setCloudChatInfo(cloudChatAppInfo);
    }
    public static void clear() {
        contextMap.remove(); //本地線程的remove方法極其重要,注意每次給它使用之后一定要調(diào)用remove清理,防止內(nèi)存泄露。
    }
}

我們發(fā)現(xiàn)上例代碼中進(jìn)行了查庫的操作:

DingAppInfo appInfo = SpringContextHolder.getBean(DingAppInfoService.class).findAppInfo(APP_CODE);

這里是不是就解決了最初我們的問題?

實(shí)現(xiàn)了擴(kuò)展applicationContextAware接口,解決靜態(tài)方法無法獲取bean實(shí)例調(diào)用其方法的問題?

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

相關(guān)文章

  • Linux 下通過 java 命令啟動(dòng) jar 包常見方式小結(jié)

    Linux 下通過 java 命令啟動(dòng) jar 包常見方式小結(jié)

    這篇文章主要介紹了Linux 下通過 java 命令啟動(dòng) jar 包常見方式小結(jié),后臺(tái)啟動(dòng)jar包命令大致有五種,每種方式結(jié)合代碼給大家介紹的非常詳細(xì),感興趣的朋友一起看看吧
    2023-12-12
  • 200行Java代碼如何實(shí)現(xiàn)依賴注入框架詳解

    200行Java代碼如何實(shí)現(xiàn)依賴注入框架詳解

    依賴注入對(duì)大家來說應(yīng)該都不陌生,下面這篇文章主要給大家介紹了關(guān)于利用200行Java代碼如何實(shí)現(xiàn)依賴注入框架的相關(guān)資料,文中通過示例代碼介紹的非常詳細(xì),需要的朋友可以參考借鑒,下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧
    2018-05-05
  • 如何從request中獲取body的數(shù)據(jù)

    如何從request中獲取body的數(shù)據(jù)

    這篇文章主要介紹了如何從request中獲取body的數(shù)據(jù)問題,具有很好的參考價(jià)值,希望對(duì)大家有所幫助,如有錯(cuò)誤或未考慮完全的地方,望不吝賜教
    2023-11-11
  • SpringBoot實(shí)用小技巧之如何動(dòng)態(tài)設(shè)置日志級(jí)別

    SpringBoot實(shí)用小技巧之如何動(dòng)態(tài)設(shè)置日志級(jí)別

    這篇文章主要給大家介紹了關(guān)于SpringBoot實(shí)用小技巧之如何動(dòng)態(tài)設(shè)置日志級(jí)別的相關(guān)資料,文中通過示例代碼介紹的非常詳細(xì),對(duì)大家學(xué)習(xí)或者使用SpringBoot具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面來一起學(xué)習(xí)學(xué)習(xí)吧
    2019-04-04
  • feign調(diào)用第三方接口,編碼定義GBK,響應(yīng)中文亂碼處理方式

    feign調(diào)用第三方接口,編碼定義GBK,響應(yīng)中文亂碼處理方式

    這篇文章主要介紹了feign調(diào)用第三方接口,編碼定義GBK,響應(yīng)中文亂碼處理方式,具有很好的參考價(jià)值,希望對(duì)大家有所幫助,如有錯(cuò)誤或未考慮完全的地方,望不吝賜教
    2024-01-01
  • 初識(shí)MyBatis及基本配置和執(zhí)行

    初識(shí)MyBatis及基本配置和執(zhí)行

    這篇文章主要介紹了初識(shí)MyBatis的基本知識(shí),文中給大家提到了mybatis基本配置和執(zhí)行過程,需要的朋友可以參考下
    2017-11-11
  • java鎖機(jī)制ReentrantLock源碼實(shí)例分析

    java鎖機(jī)制ReentrantLock源碼實(shí)例分析

    這篇文章主要為大家介紹了java鎖機(jī)制ReentrantLock源碼實(shí)例分析,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪
    2022-10-10
  • SpringBoot項(xiàng)目的測(cè)試類實(shí)例解析

    SpringBoot項(xiàng)目的測(cè)試類實(shí)例解析

    這篇文章主要介紹了SpringBoot項(xiàng)目的測(cè)試類實(shí)例解析,文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下
    2019-12-12
  • jdk源碼閱讀Collection詳解

    jdk源碼閱讀Collection詳解

    這篇文章主要介紹了jdk源碼閱讀Collection詳解,具有一定借鑒價(jià)值,需要的朋友可以參考下
    2017-12-12
  • Java實(shí)現(xiàn)網(wǎng)絡(luò)文件下載以及下載到指定目錄

    Java實(shí)現(xiàn)網(wǎng)絡(luò)文件下載以及下載到指定目錄

    在Spring框架中,StreamUtils和FileCopyUtils兩個(gè)工具類提供了方便的文件下載功能,它們都屬于org.springframework.util包,可以通過簡(jiǎn)單的方法調(diào)用實(shí)現(xiàn)文件流的復(fù)制和下載,這些工具類支持多種參數(shù)傳遞,涵蓋了文件下載的多種場(chǎng)景
    2024-09-09

最新評(píng)論