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時加載它.