1. 程式人生 > >詳解Java Web專案啟動執行順序

詳解Java Web專案啟動執行順序

一、web.xml配置節點簡介

(1) context-param

格式定義

<context-param>
  <param-name>contextConfigLocation</param-name>
  <param-value>classpath:spring/spring-mybatis.xml</param-value>
</context-param>

作用:

  1. 該元素用來宣告應用範圍(整個WEB專案)內的上下文初始化引數。
  2. param-name 設定上下文的引數名稱。必須是唯一名稱
  3. param-value 設定的引數名稱的值,這裡的例子是指定spring配置檔案的位置

(2) listener

格式定義

//listen-class 指定監聽類,該類繼承ServletContextListener 包含初始化方法contextInitialized(ServletContextEvent event) 和銷燬方法contextDestoryed(ServletContextEvent event)
<listener>
    <listener-class>org.springframework.web.context.ContextLoaderListener</listener-class>
</listener>

作用:該元素用來註冊一個監聽器類。可以收到事件什麼時候發生以及用什麼作為響應的通知。事件監聽程式在建立、修改和刪除會話或servlet環境時得到通知。常與context-param聯合使用。

(3) filter

格式定義

<filter>
  <filter-name>CharacterEncodingFilter</filter-name>
  <filter-class>org.springframework.web.filter.CharacterEncodingFilter</filter-class>
  <init-param>
    <param-name>encoding</param-name>
    <param-value>utf-8</param-value>
  </init-param>
</filter>
<filter-mapping>
  <filter-name>CharacterEncodingFilter</filter-name>
  <url-pattern>/*</url-pattern>
</filter-mapping>

作用:用於指定WEB容器的過濾器, filter能夠在一個請求到達servlet之前預處理使用者請求,也可以在離開servlet時處理http響應;在執行servlet之前,首先執行filter程式,併為之做一些預處理工作;根據程式需要修改請求和響應;在servlet被呼叫之後截獲servlet的執行。

(4)servlet 

- 格式定義

//配置Spring MVC,指定處理請求的Servlet,有兩種方式:
//1. 預設查詢MVC配置檔案的地址是:/WEB-INF/${servletName}-servlet.xml
//2. 可以通過配置修改MVC配置檔案的位置,需要在配置DispatcherServlet時指定MVC配置檔案的位置。
//這裡使用的是第二種方式

<!-- Springmvc的核心控制器 -->
  <servlet>
    <servlet-name>dispatchServlet</servlet-name>
    <servlet-class>org.springframework.web.servlet.DispatcherServlet</servlet-class>
    <init-param>
      <param-name>contextConfigLocation</param-name>
      <param-value>classpath:spring/springmvc.xml</param-value>
    </init-param>
    <load-on-startup>1</load-on-startup>
  </servlet>
  <servlet-mapping>
    <servlet-name>dispatchServlet</servlet-name>
    <url-pattern>*.shtml</url-pattern>
  </servlet-mapping>

作用:

  1. 建立並返回一個包含基於客戶請求性質的動態內容的完整的html頁面;
  2. 建立可嵌入到現有的html頁面中的一部分html頁面(html片段);
  3. 讀取客戶端發來的隱藏資料;
  4. 讀取客戶端發來的顯示資料;
  5. 與其他伺服器資源(包括資料庫和java的應用程式)進行通訊;

二、 web.xml載入過程(步驟):

  1. 啟動web專案,容器(如Tomcat、Apache)會去讀取它的配置檔案web.xml 中的兩個節點,context-param和listener。
  2. 緊接著,容器將建立一個ServletContext(又稱為:Servlet上下文),應用範圍內即整個WEB專案都能使用這個Servlet上下文。
  3. 容器將< context-param >轉化為鍵值對,並交給ServletContext。
  4. 容器建立< listener >中的類例項,即建立監聽。(備註:listener定義的類可以是自定義的類但必須需要繼承ServletContextListener)。
  5. 在監聽中會有contextInitialized(ServletContextEvent args)初始化方法,在這個方法中獲得:ServletContext = ServletContextEvent.getServletContext(); context-param的值 = ServletContext.getInitParameter(“context-param的鍵”); 在這個類中還必須有一個contextDestroyed(ServletContextEvent event) 銷燬方法。用於關閉應用前釋放資源,比如說資料庫連線的關閉。
  6. 得到這個context-param的值之後,你就可以做一些操作了。注意,這個時候你的WEB專案還沒有完全啟動完成。這個動作會比所有的Servlet都要早。換句話說,這個時候,你對 < context-param > 中的鍵值做的操作,將在你的WEB專案完全啟動之前被執行。
  7. 舉例.你可能想在專案啟動之前就開啟資料庫。那麼這裡就可以在< context-param >中設定資料庫的連線方式,在監聽類中初始化資料庫的連線。

補充知識:ServletContext,是一個全域性的儲存資訊的空間,伺服器開始,其就存在,伺服器關閉,其才釋放。request,一個使用者可有多個;session,一個使用者一個;而servletContext,所有使用者共用一個。所以,為了節省空間,提高效率,ServletContext中,要放必須的、重要的、所有使用者需要共享的執行緒又是安全的一些資訊。例如,一個購物網站,使用者要訪問商品的詳細資訊,如果放在session域,每個使用者都要訪問一遍資料庫,這樣效率太低;而放在ServletContext中,伺服器一啟動,就訪問資料庫將商品資訊放入資料庫,這樣所有使用者只需要通過上下文就能訪問到商品的資訊。

三、web.xml節點載入順序:

  • web.xml節點的載入順序與它們在web.xml中位置的先後無關,即不會因為< filter >寫在< context-param >前面就先載入< filter >。
  • 上文也提到到了,< context-param >用於對ServletContext提供鍵值對,即應用程式的上下文資訊。而listener、servlet等節點在初始化的過程中會使用到這些上下文資訊,所以最後我們得出web.xml節點的載入順序應該為:context-param->listener->filter->servlet。
  • 對於某類配置節點而言,位置的先後是有要求的。以servlet舉例,與servlet相關的配置節點是servlet-mapping,對於擁有相同配置節servlet-name的servlet和servlet-mapping來說,servlet-mapping必須在servlet後定義,否則當解析到servlet-mapping時,它的servlet-name還沒有定義。web 容器啟動時初始化每個 servlet時,是按照 servlet配置節出現的順序來初始化的。
  • 最終結論: web.xml 的載入順序是:[context-param -> listener -> filter -> servlet -> spring] ,而同類型節點之間的實際程式呼叫的時候的順序是根據對應的 mapping 的順序進行呼叫的。