先以載入spring為例子看看載入順序的作用:
Spring載入可以利用ServletContextListener 實現,也可以採用load-on-startup Servlet 實現,但比如filter 需要用到 bean ,但載入順序是: 先載入filter 後載入spring,則filter中初始化操作中的bean為null;所以,如果過濾器中要使用到 bean,此時就可以根據載入順序listener>filter>servlet,將spring 的載入 改成 Listener的方式。
利用ServletContextListener實現:
- <servlet>
- <servlet-name>context</servlet-narne>
- <servlet-class>
- org.springframework.web.context.ContextLoaderServlet
- </servlet-class>
- <load-on-startup>1</load-on-startup>
- </servlet>
<servlet>
<servlet-name>context</servlet-narne>
<servlet-class>
org.springframework.web.context.ContextLoaderServlet
</servlet-class>
<load-on-startup>1</load-on-startup>
</servlet>
採用load-on-startup Servlet 實現:
- <listener>
- <listener-class>
- org.springframework.web.context.ContextLoaderListener
- </listener-class>
- </listener>
<listener> <listener-class> org.springframework.web.context.ContextLoaderListener </listener-class> </listener>
首先可以確定的一點是載入順序與它們在web.xml裡出現的次序無關,即不會因為filter寫在listener之前而先載入fiter等等.
就<context-param>、<listener>、<filter>、<servlet>四種節點而言,啟動伺服器時的載入順序是context-param > listener > filter > servlet。
首先是:<context-param>,啟動一個WEB專案的時候,WEB容器會去讀取它的配置檔案web.xml,讀取<listener>和<context-param>兩個結點,它用於向 ServletContext 提供鍵值對,即應用程式上下文資訊。例如我們的 listener, filter 等在初始化時會用到這些上下文中的資訊。
例如:在監聽中會有contextInitialized(ServletContextEvent args)初始化方法,在這個方法中獲得:
ServletContext = ServletContextEvent.getServletContext();
context-param的值 = ServletContext.getInitParameter("context-param的鍵");
ServletContext = ServletContextEvent.getServletContext();
context-param的值 = ServletContext.getInitParameter("context-param的鍵");
它包含兩個子元素:param-name,param-value,
前者用來設定context的名字,後者用來設定其值。例如
- <context-param>
- <param-name>contextConfigLocation</param-name>
- <param-value>
- classpath*:/applicationContext.xml
- classpath*:/applicationContext-security.xml
- </param-value>
- </context-param>
<context-param> <param-name>contextConfigLocation</param-name> <param-value> classpath*:/applicationContext.xml classpath*:/applicationContext-security.xml </param-value> </context-param>
當param-value有多個值時,可以用空格或逗號隔開,或萬用字元。
其次是:listener, listener元素用來定義Listener介面,它的主要子元素為< listener-class>
< listen-class>Listener的類名稱< /listener-class>定義Listener的類名稱.例如: com.foo.hello
範例:
- <listener>
- <listener-class>
- org.springframework.web.context.ContextLoaderListener
- </listener-class>
- </listener>
<listener> <listener-class> org.springframework.web.context.ContextLoaderListener </listener-class> </listener>
接著是:filter
filter元素用來宣告filter的相關設定.filter元素除了下面介紹的的子元素之外,還包括< icon>,< display-name> ,< description>,< init-param>,其用途一樣.
< filter-name>Filter的名稱< /filter-name>定義Filter的名稱.
< filter-class>Filter的類名稱< /filter-class>定義Filter的類名稱.例如:com.foo.hello
< filter-mapping>
filter-mapping 元素的兩個主要子元素filter-name和url-pattern.用來定義Filter所對應的URL.
< filter-name>Filter的名稱< /filter-name>定義Filter的名稱.
< url-pattern>URL< /url-pattern>Filter所對應的RUL.
例如:< url-pattern>/Filter/Hello< /url-pattern>
< servlet-name>Servlet的名稱< servlet-name>定義servlet的名稱.
< dispatcher>REQUEST|INCLUDE|FORWARD|ERROR< /disaptcher>設定Filter對應的請求方式,有RQUEST,INCLUDE,FORWAR,ERROR四種,預設為REQUEST.
範例:
- <filter>
- <filter-name>
- hibernateOpenSessionInViewFilter
- </filter-name>
- <filter-class>
- org.springframework.orm.hibernate3.support.OpenSessionInViewFilter
- </filter-class>
- </filter>
- <filter-mapping>
- <filter-name>
- hibernateOpenSessionInViewFilter
- </filter-name>
- <url-pattern>*.action</url-pattern>
- <dispatcher>REQUEST</dispatcher>
- <dispatcher>FORWARD</dispatcher>
- </filter-mapping>
<filter>
<filter-name>
hibernateOpenSessionInViewFilter
</filter-name>
<filter-class>
org.springframework.orm.hibernate3.support.OpenSessionInViewFilter
</filter-class>
</filter>
<filter-mapping>
<filter-name>
hibernateOpenSessionInViewFilter
</filter-name>
<url-pattern>*.action</url-pattern>
<dispatcher>REQUEST</dispatcher>
<dispatcher>FORWARD</dispatcher>
</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
<servlet></servlet> 用來宣告一個servlet的資料,主要有以下子元素:
<servlet-name></servlet-name> 指定servlet的名稱
<servlet-class></servlet-class> 指定servlet的類名稱
<jsp-file></jsp-file> 指定web站臺中的某個JSP網頁的完整路徑
<init-param></init-param> 用來定義引數,可有多個init-param。在servlet類中通過getInitParamenter(String name)方法訪問初始化引數
<load-on-startup></load-on-startup>指定當Web應用啟動時,裝載Servlet的次序。
<servlet-mapping></servlet-mapping> 伺服器一般為servlet提供一個預設的URL:http://host/webAppPrefix/servlet/ServletName。
但是,常常會更改這個URL,以便servlet可以訪問初始化引數或更容易地處理相對URL。在更改預設URL時,使用servlet-mapping元素。
用來定義servlet所對應的URL,包含兩個子元素
<servlet-name></servlet-name> 指定servlet的名稱
<url-pattern></url-pattern> 指定servlet所對應的URL
其中的<load-on-startup></load-on-startup>當值為正數或零時:Servlet容器先載入數值小的servlet,再依次載入其他數值大的servlet. 當值為負或未定義:Servlet容器將在Web客戶首次訪問這個servlet時載入它.