亚洲激情专区-91九色丨porny丨老师-久久久久久久女国产乱让韩-国产精品午夜小视频观看

溫馨提示×

溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊×
其他方式登錄
點擊 登錄注冊 即表示同意《億速云用戶服務條款》

SpringBoot?ApplicationContextAware拓展接口怎么使用

發布時間:2023-05-17 16:59:54 來源:億速云 閱讀:127 作者:iii 欄目:編程語言

這篇文章主要介紹“SpringBoot ApplicationContextAware拓展接口怎么使用”,在日常操作中,相信很多人在SpringBoot ApplicationContextAware拓展接口怎么使用問題上存在疑惑,小編查閱了各式資料,整理出簡單好用的操作方法,希望對大家解答”SpringBoot ApplicationContextAware拓展接口怎么使用”的疑惑有所幫助!接下來,請跟著小編一起來學習吧!

ApplicationContextAware接口:

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

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

我們來看ApplicationContext對象的方法:

SpringBoot?ApplicationContextAware拓展接口怎么使用

來看看AbstractApplicationContext實現類的方法:

    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);}

這里我們可以發現 getBean()方法很眼熟,因為在最最開始學習spring時沒有用spring的腳手架創建項目,我們獲取bean的方法通常是classPathContextLoader掃描bean的xml文件解析組成ApplicationCOntext對象,再調用它的getBean方法獲取實例bean。

由此可以發現我們主要的應用途徑就是使用這個getBean的方法,那么動態的注入bean我們通過很多方法就能實現,所以這里不難想到,靜態方法中無法使用注入的bean的問題。

其次我們來復現這個問題,大家來看如下的代碼:

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);
}

我們發現在靜態的Test方法中調用注入的bean直接報錯,這里解釋一下:歸功于類的加載機制與加載順序,靜態屬性與靜態代碼塊最先加載(static靜態優先),這里加載靜態方法是沒有bean實例給你用的,自然會報錯。

如何解決?我們可以采取Spring獲取bean對象時調用getBean方法的思路,在容器加載時將spring容器的上下文進行靜態存儲:

@Component
@Lazy(value = false)
public class SpringContextHolder implements ApplicationContextAware, DisposableBean {
    /**
     * 將上下文靜態設置,在初始化組件時就進行靜態上下文的覆蓋(這個覆蓋是將遠spring容器的上下文對象引用加到我們預定設置)
     */
    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屬性未注入, 請在applicationContext.xml中定義SpringContextHolder.");
    }
    public static void pushEvent(ApplicationEvent event){
        assertContextInjected();
        applicationContext.publishEvent(event);
    }
}

這里只需要關注的是靜態成員變量ApplicationContext的定義、賦值與驗證:

    /**
     * 將上下文靜態設置,在初始化組件時就進行靜態上下文的覆蓋(這個覆蓋是將遠spring容器的上下文對象引用加到我們預定設置)
     */
    private static ApplicationContext applicationContext = null;

重寫擴展接口的方法,實現靜態上下文的覆蓋:

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

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

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

寫到這里還是不明白,這么定義一個組件,將spring上下文對象靜態覆蓋到底有何作用?

不要慌,我們來看看這個類的這個方法:

public class AppContext {
    static transient ThreadLocal<Map<String, String>> contextMap = new ThreadLocal<>();
    ......省略n行業務代碼
    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方法極其重要,注意每次給它使用之后一定要調用remove清理,防止內存泄露。
    }
}

我們發現上例代碼中進行了查庫的操作:

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

到此,關于“SpringBoot ApplicationContextAware拓展接口怎么使用”的學習就結束了,希望能夠解決大家的疑惑。理論與實踐的搭配能更好的幫助大家學習,快去試試吧!若想繼續學習更多相關知識,請繼續關注億速云網站,小編會繼續努力為大家帶來更多實用的文章!

向AI問一下細節

免責聲明:本站發布的內容(圖片、視頻和文字)以原創、轉載和分享為主,文章觀點不代表本網站立場,如果涉及侵權請聯系站長郵箱:is@yisu.com進行舉報,并提供相關證據,一經查實,將立刻刪除涉嫌侵權內容。

AI

新密市| 遂宁市| 若羌县| 綦江县| 巩留县| 宁安市| 永和县| 阳泉市| 石河子市| 高州市| 德清县| 德惠市| 伽师县| 姚安县| 伊川县| 自贡市| 交口县| 汤阴县| 泾阳县| 和龙市| 罗甸县| 正蓝旗| 阳信县| 昌黎县| 安阳县| 长汀县| 陆川县| 仁化县| 谷城县| 双鸭山市| 内乡县| 怀化市| 莆田市| 衡山县| 顺昌县| 苏州市| 宁夏| 西丰县| 庆元县| 五台县| 清水县|