SpringMVC啟動過程詳解
通過對SpringMVC啟動過程的深入研究,期望掌握Java Web容器啟動過程;掌握SpringMVC啟動過程;瞭解SpringMVC的配置檔案如何配置,為什麼要這樣配置;掌握SpringMVC是如何工作的;掌握Spring原始碼的設計和增強閱讀原始碼的技巧。
目錄
1.Web容器初始化過程
2.SpringMVC中web.xml配置
3.認識ServletContextListener
4.認識ContextLoaderListener
5.DispatcherServlet初始化(HttpServletBean • FrameworkServlet • DispatcherServlet
6.ContextLoaderListener與DispatcherServlet關係
7.DispatcherServlet的設計
8.DispatcherServlet工作原理
一、Web容器初始化過程
上圖展示了web容器初始化的過程,其官方文件給出了這樣的描述:
When a web application is deployed into a container, the following steps must be performed, in this order, before the web application begins processing client requests.
- Instantiate an instance of each event listener identified by a <listener> element in the deployment descriptor.
- For instantiated listener instances that implement ServletContextListener, call the contextInitialized() method.
- Instantiate an instance of each filter identified by a <filter> element in the deployment descriptor and call each filter instance's init() method.
- Instantiate an instance of each servlet identified by a <servlet> element that includes a <load-on-startup> element in the order defined by the load-on-startup element values, and call each servlet instance's init() method.
二、SpringMVC中web.xml的配置
上圖是擷取的web.xml中的配置,在<listener>標籤中定義了spring容器載入器;在<servlet>標籤中定義了spring前端控制器。
上圖是原始碼中介面ServletContextListener的定義,可以看到在其註釋中指明:servlet和Filter初始化前和銷燬後,都會給實現了servletContextListener介面的監聽器發出相應的通知。
上面是類ContextLoadListener的定義,它實現了上面的servletContextListener。這裡用到了代理模式,簡單的代理了ContextLoader類。ContextLoadListener類用來建立Spring application context,並且將application context註冊到servletContext裡面去
結合上面的WEB容器啟動的過程,以及介面ServletContextListener和類ContextLoadListener。我們知道:
在 Servlet API中有一個ServletContextListener介面,它能夠監聽ServletContext物件的生命週期,實際上就是監聽Web應用的生命週期。當Servlet容器啟動或終止Web應用時,會觸發ServletContextEvent事件,該事件由ServletContextListener來處理。在ServletContextListener介面中定義了處理ServletContextEvent 事件的兩個方法contextInitialized()和contextDestroyed()。
ContextLoaderListener監聽器的作用就是啟動Web容器時,自動裝配ApplicationContext的配置資訊。因為它實現了ServletContextListener這個介面,在web.xml配置了這個監聽器,啟動容器時,就會預設執行它實現的方法。由於在ContextLoaderListener中關聯了ContextLoader這個類,所以整個載入配置過程由ContextLoader來完成。
上面是initWebApplicationContext的過程,方法名稱即是其含義。方法中首先建立了WebApplicationContext,配置並且重新整理例項化整個SpringApplicationContext中的Bean。因此,如果我們的Bean配置出錯的話,在容器啟動的時候,會拋異常出來的。
綜上,ContextLoaderListener類起著至關重要的作用。它讀取web.xml中配置的context-param中的配置檔案,提前在web容器初始化前準備業務對應的Application context;將建立好的Application context放置於ServletContext中,為springMVC部分的初始化做好準備。
三、DispatchServlet初始化
在SpringMVC架構中,DispatchServlet負責請求分發,起到控制器的作用。下面詳細來解釋說明:
DispatchServlet名如其義,它的本質上是一個Servlet。從上面圖可以看到,下層的子類不斷的對HttpServlet父類進行方法擴充套件。
上圖是抽象類HttpServletBean的實現,我們知道HttpServlet有兩大核心方法:init()和service()方法。HttpServletBean重寫了init()方法,在這部分,我們可以看到其實現思路:公共的部分統一來實現,變化的部分統一來抽象,交給其子類來實現,故用了abstract class來修飾類名。此外,HttpServletBean提供了一個HttpServlet的抽象實現,使的Servlet不再關心init-param部分的賦值,讓servlet更關注於自身Bean初始化的實現。
上圖是FrameworkServlet的官方定義, 它提供了整合web javabean和spring application context的整合方案。那麼它是如何實現的呢?在原始碼中我們可以看到通過執行initWebApplicationContext()方法和initFrameworkServlet()方法實現。
DispatchServlet是HTTP請求的中央排程處理器,它將web請求轉發給controller層處理,它提供了敏捷的對映和異常處理機制。DispatchServlet轉發請求的核心程式碼在doService()方法中實現,詳細程式碼參照圖上。
上圖是DispatchServlet類和ContextLoaderListener類的關係圖。首先,用ContextLoaderListener初始化上下文,接著使用DispatchServlet來初始化WebMVC的上下文。
上圖是DispatchServlet的工作流程圖,作為HTTP請求的中央控制器,它在SpringMVC中起著分發請求的作用。下面總結了DispatchServlet設計的一些特點總結。