一、
spring的啟動過程:
1.首先,對於一個web應用,其部署在web容器中,web容器提供其一個全局的上下文環境,這個上下文就是ServletContext,其為后面的spring IoC容器提供宿主環境;
2.其 次,在web.xml中會提供有contextLoaderListener。在web容器啟動時,會觸發容器初始化事件,此時 contextLoaderListener會監聽到這個事件,其contextInitialized方法會被調用,在這個方法中,spring會初始 化一個啟動上下文,這個上下文被稱為根上下文,即WebApplicationContext,這是一個接口類,確切的說,其實際的實現類是 XmlWebApplicationContext。這個就是spring的IoC容器,其對應的Bean定義的配置由web.xml中的 context-param標簽指定。在這個IoC容器初始化完畢后,spring以WebApplicationContext.ROOTWEBAPPLICATIONCONTEXTATTRIBUTE為屬性Key,將其存儲到ServletContext中,便於獲取;
3.再 次,contextLoaderListener監聽器初始化完畢后,開始初始化web.xml中配置的Servlet,這里是DispatcherServlet,這個servlet實際上是一個標准的前端控制器,用以轉發、匹配、處理每個servlet請 求。DispatcherServlet上下文在初始化的時候會建立自己的IoC上下文,用以持有spring mvc相關的bean。在建立DispatcherServlet自己的IoC上下文時,會利用WebApplicationContext.ROOTWEBAPPLICATIONCONTEXTATTRIBUTE 先從ServletContext中獲取之前的根上下文(即WebApplicationContext)作為自己上下文的parent上下文。有了這個 parent上下文之后,再初始化自己持有的上下文。這個DispatcherServlet初始化自己上下文的工作在其initStrategies方 法中可以看到,大概的工作就是初始化處理器映射、視圖解析等。這個servlet自己持有的上下文默認實現類也是 mlWebApplicationContext。初始化完畢后,spring以與servlet的名字相關(此處不是簡單的以servlet名為 Key,而是通過一些轉換,具體可自行查看源碼)的屬性為屬性Key,也將其存到ServletContext中,以便后續使用。這樣每個servlet 就持有自己的上下文,即擁有自己獨立的bean空間,同時各個servlet共享相同的bean,即根上下文(第2步中初始化的上下文)定義的那些 bean。
二、DispatcherServlet 處理流程
在整個 Spring MVC 框架中,DispatcherServlet 處於核心位置,它負責協調和組織不同組件完成請求處理並返回響應工作。在看 DispatcherServlet 類之前,我們先來看一下請求處理的大致流程:
- Tomcat 啟動,對 DispatcherServlet 進行實例化,然后調用它的 init() 方法進行初始化,在這個初始化過程中完成了: 對 web.xml 中初始化參數的加載;建立 WebApplicationContext (SpringMVC的IOC容器);進行組件的初始化;
- 客戶端發出請求,由 Tomcat 接收到這個請求,如果匹配 DispatcherServlet 在 web.xml 中配置的映射路徑,Tomcat 就將請求轉交給 DispatcherServlet 處理;
- DispatcherServlet 從容器中取出所有 HandlerMapping 實例(每個實例對應一個 HandlerMapping 接口的實現類)並遍歷,每個 HandlerMapping 會根據請求信息,通過自己實現類中的方式去找到處理該請求的 Handler (執行程序,如Controller中的方法),並且將這個 Handler 與一堆 HandlerInterceptor (攔截器) 封裝成一個 HandlerExecutionChain 對象,一旦有一個 HandlerMapping 可以找到 Handler 則退出循環;(詳情可以看 [Java]SpringMVC工作原理之二:HandlerMapping和HandlerAdpater 這篇文章)
- DispatcherServlet 取出 HandlerAdapter 組件,根據已經找到的 Handler,再從所有 HandlerAdapter 中找到可以處理該 Handler 的 HandlerAdapter 對象;
- 執行 HandlerExecutionChain 中所有攔截器的 preHandler() 方法,然后再利用 HandlerAdapter 執行 Handler ,執行完成得到 ModelAndView,再依次調用攔截器的 postHandler() 方法;
- 利用 ViewResolver 將 ModelAndView 或是 Exception(可解析成 ModelAndView)解析成 View,然后 View 會調用 render() 方法再根據 ModelAndView 中的數據渲染出頁面;
- 最后再依次調用攔截器的 afterCompletion() 方法,這一次請求就結束了
http://www.cnblogs.com/tengyunhao/p/7518481.html
