Spring ApplicationContext講解與獲得
ApplicationContext
是Spring的核心,Context我們通常解釋為上下文環境,我想用“容器”來表述它更容易理解一些,ApplicationContext則是“應
用的容器”了:P,Spring把Bean放在這個容器中,在需要的時候,用getBean方法取出,雖然我沒有看過這一部分的原始碼,但我想它應該是一
個類似Map的結構。
在Web應用中,我們會用到WebApplicationContext,WebApplicationContext繼承自
ApplicationContext,先讓我們看看在Web應用中,怎麼初始化WebApplicationContext,在web.xml中定
義:
<context-param> <param-name>contextConfigLocation</param-name> <param-value>/WEB-INF/applicationContext.xml</param-value> </context-param> <listener> <listener-class>org.springframework.web.context.ContextLoaderListener</listener-class> </listener> <!-- OR USE THE CONTEXTLOADERSERVLET INSTEAD OF THE LISTENER <servlet> <servlet-name>context</servlet-name> <servlet-class>org.springframework.web.context.ContextLoaderServlet</servlet-class> <load-on-startup>1</load-on-startup> </servlet> -->
可以看出,有兩種方法,一個是用ContextLoaderListener這個Listerner,另一個是ContextLoaderServlet
這個Servlet,這兩個方法都是在web應用啟動的時候來初始化WebApplicationContext,我個人認為Listerner要比
Servlet更好一些,因為Listerner監聽應用的啟動和結束,而Servlet得啟動要稍微延遲一些,如果在這時要做一些業務的操作,啟動的前
後順序是有影響的。
那麼在ContextLoaderListener和ContextLoaderServlet中到底做了什麼呢?
以ContextLoaderListener為例,我們可以看到
public void contextInitialized(ServletContextEvent event) {
this.contextLoader = createContextLoader();
this.contextLoader.initWebApplicationContext(event.getServletContext());
}
protected ContextLoader createContextLoader() {
return new ContextLoader();
}
ContextLoader是一個工具類,用來初始化WebApplicationContext,其主要方法就是initWebApplicationContext,我們繼續追蹤initWebApplicationContext這個方法(具體程式碼我不貼出,大家可以看Spring中的原始碼),我們發現,原來ContextLoader是把WebApplicationContext(XmlWebApplicationContext是
預設實現類)放在了ServletContext中,ServletContext也是一個“容器”,也是一個類似Map的結構,而
WebApplicationContext在ServletContext中的KEY就是
WebApplicationContext.ROOT_WEB_APPLICATION_CONTEXT_ATTRIBUTE,我們如果要使用
WebApplicationContext則需要從ServletContext取出,Spring提供了一個
WebApplicationContextUtils類,可以方便的取出WebApplicationContext,只要把ServletContext傳入就可以了。
上面我們介紹了WebApplicationContext在Servlet容器中初始化的原理,一般的Web應用就可以輕鬆的使用了,但是,隨著
Struts的廣泛應用,把Struts和Spring整個起來,是一個需要面對的問題,Spring本身也提供了Struts的相關類,主要使用的有
org.springframework.web.struts.ActionSupport,我們只要把自己的Action繼承自
ActionSupport,就是可以呼叫ActionSupport中getWebApplicationContext()的方法取出
WebApplicationContext,但這樣一來在Action中,需要取得業務邏輯的地方都要getBean,看上去不夠簡潔,所以
Spring又提供了另一個方法,用org.springframework.web.struts.ContextLoaderPlugIn,這是一個
Struts的Plug,在Struts啟動時載入,對於Action,可以像管理Bean一樣來管理,在struts-config.xml中
Action的配置變成類似下面的樣子
<action attribute="aForm" name="aForm" path="/aAction" scope="request" type="org.springframework.web.struts.DelegatingActionProxy"> <forward name="forward" path="forward.jsp" /> </action>
注意type變成了org.springframework.web.struts.DelegatingActionProxy,之後我們需要建立action-servlet.xml
這樣的檔案,action-servlet.xml
符合Spring的spring-beans.dtd標準,在裡面定義類似下面的
<bean name="/aAction" class="com.web.action.Aaction" singleton="false"> <property name="businessService"> <ref bean="businessService"/> </property> </bean>
com.web.action.Aaction是Action的實現類,businessService是需要的業務邏輯,Spring會把
businessService注入到Action中,在Action中只要寫businessService的get和set方法就可以了,還有一
點,action的bean是singleton="false",即每次新建一個例項,這也解決了Struts中Action的執行緒同步問題,具體過程
是當使用者做“/aAction”的HTTP請求(當然應該是“/aAction.do”),Struts會找到這個Action的對應類
org.springframework.web.struts.DelegatingActionProxy,DelegatingActionProxy
是個代理類,它會去找action-servlet.xml
檔案中“/aAction”對應的真正實現類,然後把它例項化,同時把需要的業務物件注入,然後執行Action的execute方法。
使用了ContextLoaderPlugIn,在struts-config.xml中變成類似這樣配置
<plug-in className="org.springframework.web.struts.ContextLoaderPlugIn"> <set-property property="contextConfigLocation" value="/WEB-INF/applicationContext.xml,/WEB-INF/action-servlet.xml " /> </plug-in>
而在web.xml中不再需要ContextLoaderListener或是ContextLoaderServlet。
說到這裡不知道大家會不會有這樣的問題,如果使用ContextLoaderPlugIn,如果我們有些程式是脫離Struts的Action環境,我們
怎麼處理,比如我們要自定義標記庫,在標記庫中,我們需要呼叫Spring管理的業務層邏輯物件,這時候我們就很麻煩,因為只有在action中動態注入
業務邏輯,其他我們似乎不能取得Spring的WebApplicationContext。
別急,我們還是來看一下ContextLoaderPlugIn的原始碼(原始碼不再貼出),我們可以發現,原來ContextLoaderPlugIn仍然
是把WebApplicationContext放在ServletContext中,只是這個KEY不太一樣了,這個KEY值為
ContextLoaderPlugIn.SERVLET_CONTEXT_PREFIX+ModuleConfig.getPrefix()(具體請查
看原始碼),這下好了,我們知道了WebApplicationContext放在哪裡,只要我們在Web應用中能夠取到ServletContext也
就能取到WebApplicationContext了:)
相關文章
- Spring - 獲取ApplicationContext的幾種方式SpringAPPContext
- Spring BeanFactory與ApplicationContext 介紹SpringBeanAPPContext
- spring beanFactory與ApplicationContext區別SpringBeanAPPContext
- Spring之BeanFactory與ApplicationConText區別SpringBeanAPPContext
- Spring之ApplicationContextSpringAPPContext
- 在Spring boot中通過ApplicationContext獲取bean失敗Spring BootAPPContextBean
- Spring原始碼系列(補充):詳解ApplicationContextSpring原始碼APPContext
- spring applicationContext.xml 配置檔案 詳解SpringAPPContextXML
- Spring中獲得dataSurce物件的方法Spring物件
- Spring(ApplicationContext&BeanFactory)SpringAPPContextBean
- Spring BeanFactory和ApplicationContextSpringBeanAPPContext
- spring的BeanFactory和ApplicationContextSpringBeanAPPContext
- 深入Spring Security-獲取認證機制核心原理講解Spring
- spring applicationContext家族層級梳理SpringAPPContext
- Spring核心系列之ApplicationContextSpringAPPContext
- Spring配置初始化ApplicationContextSpringAPPContext
- Spring Boot整合 Geodesy講解Spring Boot
- ASPECT中如何獲得SPRING所管理的事務.?Spring
- 深入Spring Security魔幻山谷-獲取認證機制核心原理講解(新版)Spring
- 解決Spring Boot測試方法Failed to load ApplicationContext問題Spring BootAIAPPContext
- Spring入門(七):Spring Profile使用講解Spring
- ApplicationContext 與 BeanFactory 區別APPContextBean
- Spring 通過Spring容器獲得資料來源物件並改進Spring物件
- ApplicationContext中getBean詳解APPContextBean
- 工廠模式講解, 引入Spring IOC模式Spring
- Spring4.0MVC學習資料,ApplicationContext中的方法詳解(三)SpringMVCAPPContext
- git概念與指令講解Git
- 解決:Failed to load ApplicationContextAIAPPContext
- spring中的applicationContext的應用範圍SpringAPPContext
- Spring @Conditional註解 詳細講解及示例Spring
- 敏捷聯盟Gordon Pask獎獲得者講“測試驅動開發”(TDD)敏捷Go
- Spring中的事件講解(Application Event)Spring事件APP
- 死磕Spring之IoC篇 - Spring 應用上下文 ApplicationContextSpringAPPContext
- php獲得時間PHP
- Javascript之BOM與DOM講解JavaScript
- Spring IOC原始碼研究筆記(2)——ApplicationContext系列Spring原始碼筆記APPContext
- 【Spring註解驅動開發】二狗子讓我給他講講@EnableAspectJAutoProxy註解Spring
- Spring Boot註解@Transactional結合實際例子講解Spring Boot