獲取Spring的上下文環境ApplicationContext的最簡單方式

獲取Spring上下文環境ApplicationContext

Web項目中發現有人如此獲得Spring的上下環境:

public class SpringUtil {
       public static ApplicationContext context = new ClassPathXmlApplicationContext("applicationContext.xml");
       public static Object getBean(String serviceName){
             return context.getBean(serviceName);
       }
}

在web項目中這種方式非常不可取!!!

分析:

首先,主要意圖就是獲得Spring上下文;

其次,有瞭Spring上下文,希望通過getBean()方法獲得Spring管理的Bean的對象;

最後,為瞭方便調用,把上下文定義為static變量或者getBean方法定義為static方法;

但是,在web項目中,系統一旦啟動,web服務器會初始化Spring的上下文的,我們可以很優雅的獲得Spring的ApplicationContext對象。

如果使用

new ClassPathXmlApplicationContext("applicationContext.xml");

相當於重新初始化一遍!!!!

也就是說,重復做啟動時候的初始化工作,第一次執行該類的時候會非常耗時!!!!!

正確的做法是:

@Component
public class SpringContextUtil implements ApplicationContextAware {
         private static ApplicationContext applicationContext; // Spring應用上下文環境
         /*
          * 實現瞭ApplicationContextAware 接口,必須實現該方法;
          *通過傳遞applicationContext參數初始化成員變量applicationContext
          */
         public void setApplicationContext(ApplicationContext applicationContext) throws BeansException {
               SpringContextUtil.applicationContext = applicationContext;
         } 
         public static ApplicationContext getApplicationContext() {
                return applicationContext;
         }
          @SuppressWarnings("unchecked")
          public static <T> T getBean(String name) throws BeansException {
                     return (T) applicationContext.getBean(name);
           }
}

註意:

這個地方使用瞭Spring的註解@Component,如果不是使用annotation的方式,而是使用xml的方式管理Bean,記得寫入配置文件

<bean id="springContextUtil" class="com.ecdatainfo.util.SpringContextUtil" singleton="true" />

其實

ApplicationContext context = new ClassPathXmlApplicationContext("applicationContext.xml");

這種方式獲取Sping上下文環境,最主要是在測試環境中使用,比如寫一個測試類,系統不啟動的情況下手動初始化Spring上下文再獲取對象!

Spring上下文(ApplicationContext)理解

什麼是Spring應用上下文呢???

什麼是應用上下文呢,應用上下文即是Spring容器的一種抽象化表述;而我們常見的ApplicationContext本質上說就是一個維護Bean定義以及對象之間協作關系的高級接口。

Spring的核心是容器,而容器並不唯一,框架本身就提供瞭很多個容器的實現,大概分為兩種類型:一種是不常用的BeanFactory,這是最簡單的容器,隻能提供基本的DI功能;還有一種就是繼承瞭BeanFactory後派生而來的應用上下文,其抽象接口也就是上面提到的的ApplicationContext,它能提供更多企業級的服務,例如解析配置文本信息等等,這也是應用上下文實例對象最常見的應用場景。

有瞭上下文對象,我們就能向容器註冊需要Spring管理的對象瞭。對於上下文抽象接口,Spring也為我們提供瞭多種類型的容器實現,供我們在不同的應用場景選擇——

  • ① AnnotationConfigApplicationContext:從一個或多個基於java的配置類中加載上下文定義,適用於java註解的方式;
  • ② ClassPathXmlApplicationContext:從類路徑下的一個或多個xml配置文件中加載上下文定義,適用於xml配置的方式;
  • ③ FileSystemXmlApplicationContext:從文件系統下的一個或多個xml配置文件中加載上下文定義,也就是說系統盤符中加載xml配置文件;
  • ④ AnnotationConfigWebApplicationContext:專門為web應用準備的,適用於註解方式;
  • ⑤ XmlWebApplicationContext:從web應用下的一個或多個xml配置文件加載上下文定義,適用於xml配置方式。

備註:

通過.class文件獲取bean的方法:

public abstract class AbstractCreateRoomHandler implements ApplicationContextAware {
    //存放商傢和處理器的映射關系
    private static final Map<Integer, Class<? extends AbstractCreateRoomHandler>> TYPE_HANDLER = new HashMap<>();
    private static ApplicationContext applicationContext;
    static {
        TYPE_HANDLER.put(BusinessCodeEnum.TUSE.getBusinessCode(), TuSeCreateRoomHandler.class);
        //todo 新的商傢類型添加到此處
    }
    public static AbstractCreateRoomHandler getCreateRoomHandler(Integer productType) {
        Class<? extends AbstractCreateRoomHandler> clazz = TYPE_HANDLER.getOrDefault(productType, TuSeCreateRoomHandler.class);
        return applicationContext.getBean(clazz);
    }
    @Override
    public void setApplicationContext(ApplicationContext applicationContext) throws BeansException {
        this.applicationContext = applicationContext;
    }

以上為個人經驗,希望能給大傢一個參考,也希望大傢多多支持WalkonNet。

推薦閱讀: