posts - 3,  comments - 0,  trackbacks - 0
            2008年12月30日


          ApplicationContext是Spring的核心,Context我們通常解釋為上下文環(huán)境,我想用“容器”

          來表述它更容易理解一些,ApplicationContext則是“應(yīng)用的容器”了:P,Spring把Bean放在

          這個容器中,在需要的時候,用getBean方法取出,雖然我沒有看過這一部分的源代碼,但我

          想它應(yīng)該是一個類似Map的結(jié)構(gòu)。
          在Web應(yīng)用中,我們會用到WebApplicationContext,WebApplicationContext繼承自

          ApplicationContext,先讓我們看看在Web應(yīng)用中,怎么初始化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應(yīng)用啟動的時候來初始化

          WebApplicationContext,我個人認為Listerner要比Servlet更好一些,因為Listerner監(jiān)聽應(yīng)

          用的啟動和結(jié)束,而Servlet得啟動要稍微延遲一些,如果在這時要做一些業(yè)務(wù)的操作,啟動

          的前后順序是有影響的。

          那么在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,我們繼續(xù)追蹤initWebApplicationContext這個方法(具體代碼

          我不貼出,大家可以看Spring中的源碼),我們發(fā)現(xiàn),原來ContextLoader是把

          WebApplicationContext(XmlWebApplicationContext是默認實現(xiàn)類)放在了ServletContext

          中,ServletContext也是一個“容器”,也是一個類似Map的結(jié)構(gòu),而WebApplicationContext

          在ServletContext中的KEY就是

          WebApplicationContext.ROOT_WEB_APPLICATION_CONTEXT_ATTRIBUTE,我們?nèi)绻褂?/p>

          WebApplicationContext則需要從ServletContext取出,Spring提供了一個

          WebApplicationContextUtils類,可以方便的取出WebApplicationContext,只要把

          ServletContext傳入就可以了。

          上面我們介紹了WebApplicationContext在Servlet容器中初始化的原理,一般的Web應(yīng)用就可

          以輕松的使用了,但是,隨著Struts的廣泛應(yīng)用,把Struts和Spring整個起來,是一個需要面

          對的問題,Spring本身也提供了Struts的相關(guān)類,主要使用的有

          org.springframework.web.struts.ActionSupport,我們只要把自己的Action繼承自

          ActionSupport,就是可以調(diào)用ActionSupport中g(shù)etWebApplicationContext()的方法取出

          WebApplicationContext,但這樣一來在Action中,需要取得業(yè)務(wù)邏輯的地方都要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的實現(xiàn)類,businessService是需要的業(yè)務(wù)邏輯,Spring會

          把businessService注入到Action中,在Action中只要寫businessService的get和set方法就可

          以了,還有一點,action的bean是singleton="false",即每次新建一個實例,這也解決了

          Struts中Action的線程同步問題,具體過程是當用戶做“/aAction”的HTTP請求(當然應(yīng)該是

          “/aAction.do”),Struts會找到這個Action的對應(yīng)類

          org.springframework.web.struts.DelegatingActionProxy,DelegatingActionProxy是個代

          理類,它會去找action-servlet.xml文件中“/aAction”對應(yīng)的真正實現(xiàn)類,然后把它實例化

          ,同時把需要的業(yè)務(wù)對象注入,然后執(zhí)行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環(huán)境,我們怎么處理,比如我們要自定義標記庫,在標記庫中,我們

          需要調(diào)用Spring管理的業(yè)務(wù)層邏輯對象,這時候我們就很麻煩,因為只有在action中動態(tài)注入

          業(yè)務(wù)邏輯,其他我們似乎不能取得Spring的WebApplicationContext。

          別急,我們還是來看一下ContextLoaderPlugIn的源碼(源碼不再貼出),我們可以發(fā)現(xiàn),原

          來ContextLoaderPlugIn仍然是把WebApplicationContext放在ServletContext中,只是這個

          KEY不太一樣了,這個KEY值為

          ContextLoaderPlugIn.SERVLET_CONTEXT_PREFIX+ModuleConfig.getPrefix()(具體請查看源

          代碼),這下好了,我們知道了WebApplicationContext放在哪里,只要我們在Web應(yīng)用中能夠

          取到ServletContext也就能取到WebApplicationContext了:)

          Spring是一個很強大的框架,希望大家在使用過程中不斷的深入,了解其更多的特性,我在這

          里拋磚引玉,有什么不對的地方,請大家指出。

          posted @ 2009-01-07 10:18 jk 閱讀(1372) | 評論 (0)編輯 收藏
          服務(wù)器啟動以后,(Servlet容器啟動)創(chuàng)建了許多對象,如 servlet, filter, listener,spring等等 那么如何使用這些對象呢?  下面介紹在Servlet(或者Filter,或者Listener)中使用spring的IOC容器
          默認情況下Servlet容器創(chuàng)建spring容器對象,注入到servletContext中,servletContext對象又是注入到session對象中,session對象又是注入到request對象中,request對象又是注入到servlet對象中,(其實不是很標準的注入,是傳參數(shù),或者對屬性直接付值)。層層依賴可以得到spring容器對象。

          WebApplicationContext webApplicationContext = WebApplicationContextUtils.getWebApplicationContext(request.getSession().getServletContext()); 

                  所以可以直接在ServletContext取出WebApplicationContext 對象:

          WebApplicationContext webApplicationContext = (WebApplicationContext) servletContext.getAttribute(WebApplicationContext.ROOT_WEB_APPLICATION_CONTEXT_ATTRIBUTE);

          事實上WebApplicationContextUtils.getWebApplicationContext方法就是使用上面的代碼實現(xiàn)的,建議使用上面上面的靜態(tài)方法 


          注意:在使用webApplicationContext.getBean("ServiceName")的時候,前面強制轉(zhuǎn)化要使用接口,如果使用實現(xiàn)類會報類型轉(zhuǎn)換錯誤。如:
          LUserService userService = (LUserService) webApplicationContext.getBean("userService");
          posted @ 2008-12-30 14:01 jk 閱讀(418) | 評論 (0)編輯 收藏
          主站蜘蛛池模板: 将乐县| 运城市| 博乐市| 红原县| 林周县| 富民县| 昭觉县| 古蔺县| 沙坪坝区| 安康市| 贵港市| 新余市| 杂多县| 商南县| 曲阜市| 城市| 象山县| 道孚县| 香港| 色达县| 澜沧| 鄱阳县| 花垣县| 扶风县| 石家庄市| 宽甸| 安国市| 治多县| 云龙县| 高州市| 赣州市| 阿巴嘎旗| 宣武区| 革吉县| 宜宾县| 正镶白旗| 长治市| 新建县| 崇文区| 保靖县| 贡山|