web.xml文件中配置( listenr, filter,servlet, spring)的加載順序 研究(轉)


 

web.xml 文件中一般包括 servlet, spring, filter, listenr的配置。那么他們是按照一個什么順序加載呢?

加載順序會影響對spring bean 的調用。

    比如filter 需要用到 bean ,但是加載順序是 先加載filter 后加載spring,則filter中初始化操作中的bean為null;

首先可以肯定 加載順序與他們在web.xml 文件中的先后順序無關。

web.xml 中 listener 和 serverlet 的加載順序為 先 listener 后serverlet

最終得出結果:先 listener >> filter >> servlet >> spring

所以,如果過濾器中要使用到 bean,可以將spring 的加載 改成 Listener的方式

<listener>
        <listener-class>
            org.springframework.web.context.ContextLoaderListener
        </listener-class>
    </listener>


搞定!

關於他們的內部執行順序,也需要注意,如下面文章中遇到的問題

web.xml的filter執行順序導致的亂碼,切記!
2008-05-02 01:38
發現引起bug的原因是web.xml的下面幾行:
    <filter-mapping>
        <filter-name>SecurityFilter</filter-name>
        <url-pattern>*.do</url-pattern>
    </filter-mapping>

    <filter-mapping>
        <filter-name>CharacterEncoding</filter-name>
        <url-pattern>*.do</url-pattern>
    </filter-mapping>

    <filter-mapping>
        <filter-name>CharacterEncoding</filter-name>
        <url-pattern>*.jsp</url-pattern>
    </filter-mapping>

根據servlet2.3規范filter執行是按照web.xml配置的filter-mapping先后順序進行執行,所以上面的配置會導致遇見*.do的url請求,先進行SecurityFilter的過濾器處理,這時候沒有做編碼處理,已經是亂碼,到下面的filter處理時已經時亂碼,再做編碼處理已經沒有用處。

修正方式,調整filter-mapping順序,如下:

    <filter-mapping>
        <filter-name>CharacterEncoding</filter-name>
        <url-pattern>*.do</url-pattern>
    </filter-mapping>

    <filter-mapping>
        <filter-name>CharacterEncoding</filter-name>
        <url-pattern>*.jsp</url-pattern>
    </filter-mapping>

    <filter-mapping>
        <filter-name>SecurityFilter</filter-name>
        <url-pattern>*.do</url-pattern>
    </filter-mapping>

本文來自CSDN博客,轉載請標明出處:http://blog.csdn.net/wayfoon322/archive/2008/05/08/2418011.aspx

----------------------------------------------------------------------------------------------------------

首先可以確定的一點是, 不同類型節點的加載順序與它們在web.xml里出現的次序無關,即不會因為filter寫在listener之前而先加載fiter.

    就<context-param>、<listener>、<filter>、<servlet>四種節點而言,啟動服務器時的加載順序是context-param --> listener --> filter --> servlet

    首先是<context-param>,啟動一個WEB項目的時候,WEB容器會讀取配置描述符文件(web.xml)中的<context-param>節點信息。<context-param>它用於向 ServletContext 提供鍵值對,即應用程序上下文信息。在初始化listener, filter和servlet時到可能會用到這些信息。

    例如在監聽器(listener)中有一個contextInitialized(ServletContextEvent contextEvent)初始化方法,在這個方法中可以通過 contextEvent .getServletContext() 可以獲得ServletContext對象,然后再從ServletContext中通過getInitParameter(String)方法獲取<context-param>的鍵值: 

1 ServletContext servletContext = contextEvent.getServletContext(); 
2 Object context-param的值 = servletContext .getInitParameter("context-param的鍵");

<context-param>包含兩個子元素,分別是<param-name>和<param-value>。 前者用來設置context的名字,后者用來設置其值。

當param-value有多個值時, 用逗號隔開。 <param-value>里也可以使用通配符,比如:

1 <strong> <context-param>  
2        <param-name>contextConfigLocation</param-name>  
3        <param-value>  
4            classpath*:/applicationContext.xml,classpath*:/applicationContext-security.xml  
5        </param-value>  
6 </context-param> </strong>

接着初始化listener, 主要子元素為< listener-class>,如: 

1 <listener>  
2     <listener-class>  
3         org.springframework.web.context.ContextLoaderListener  
4     </listener-class>  
5 </listener>  

然后是filter。 filter元素主要包括<filter-name>,<filter-class>和<init-param>, 一個典型的<filter>節點聲明如下:

1 <filter>
2     <filter-name>Cache Filter - Friendly</filter-name>
3     <filter-class>com.liferay.portal.servlet.filters.cache.CacheFilter</filter-class>
4     <init-param>
5         <param-name>pattern</param-name>
6         <param-value>0</param-value>
7     </init-param>
8 </filter>

filter-mapping用來定義filter所對應的url匹配模式,它有兩個子元素,分別是<filter-name>和<url-pattern>,一個典型的<filter-mapping>節點如下: 

1 <filter-mapping>
2     <filter-name>Cache Filter - Layout</filter-name>
3     <url-pattern>/c/portal/layout</url-pattern>
4 </filter-mapping>
 

注意: 對於同一類型的配置節而言,與它們出現的順序是有關的。以<filter>為例,web.xml中當然可以定義多個<filter>,與<filter>相關的一個配置節是<filter-mapping>,這里一定要注意,對於擁有相同<filter-name>的<filter>和<filter-mapping>配置節而言,<filter-mapping>必須出現在<filter>之后,否則當解析到<filter-mapping>時,它所對應的<filter-name>還未定義。web 容器啟動時初始化每個<filter>時,是按照<filter>配置節出現的順序來初始化的,當請求資源匹配多個<filter-mapping>時,<filter>攔截資源是按照<filter-mapping>配置節出現的順序來依次調用doFilter() 方法的。

最后初始化servlet。和filter一樣,servlet也分<servlet>和<servlet-mapping>兩部分聲明。一個典型的servlet聲明如下:

 

 1 <servlet>
 2     <servlet-name>Friendly URL Servlet - Private User</servlet-name>
 3     <servlet-class>com.liferay.portal.servlet.FriendlyURLServlet</servlet-class>
 4     <init-param>
 5         <param-name>user</param-name>
 6         <param-value>true</param-value>
 7     </init-param>
 8     <load-on-startup>2</load-on-startup>
 9 </servlet>
10 <servlet-mapping>
11         <servlet-name>Friendly URL Servlet - Private User</servlet-name>
12         <url-pattern>/user/*</url-pattern>
13 </servlet-mapping>

其中的<load-on-startup>值為正數或零時:Servlet容器先加載數值小的servlet,再依次加載其他數值大的servlet. 當值為負或未定義:Servlet容器將在Web客戶首次訪問這個servlet時加載它. 


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM