前言
其實一年前就想系統地記錄下自己閱讀spring源碼的收穫,搞一個深入淺出spring的系列文章,但是因爲工作原因,遲遲沒有下筆。今天終於可以開始自己一年前的計劃了,言歸正傳,首先開始spring mvc相關的內容。
文本主要從流程角度介紹spring mvc的整個工作原理,對spring mvc有一個整體的瞭解,具體實現細節不在本文的介紹範疇中。
概述
Spring MVC 是目前主流的Wew MVC框架之一。兩個核心點:
- 處理器映射:選擇使用哪個控制器來處理請求
- 請求參數解析 + 返回數據解析
- 視圖解析器:選擇結果應該如何渲染
以下分析基於spring 5.0.4
版本源碼
運行原理
概述
1. 首先用戶發送請求,DispatcherServlet
實現了Servlet
接口,整個請求處理流:HttpServlet.service -> FrameworkServlet.doGet -> FrameworkServlet.processRequest -> DispatcherServlet.doService -> DispatcherServlet.doDispatch
。 doDispatch(HttpServletRequest request, HttpServletResponse response)
方法即爲整個spring mvc的處理流程。
2. 獲取url請求對應的處理方法,遍歷handlerMappings列表,獲取對象HandlerExecutionChain
(包含一個處理器 handler 如HandlerMethod 對象、多個 HandlerInterceptor 攔截器對象)。此處的handlerMappings列表爲上下文中所有HandlerMapping接口的實現類(如圖中列舉了4個),遍歷handlerMappings列表,針對每個handlerMapping試圖獲取HandlerExecutionChain,一旦成功(不爲null),即返回。這部分的詳細分析見【[深入淺出spring】Spring MVC 流程解析 – HanndlerMapping][1]
3. 獲取對應的 HandlerAdapter,HandlerAdapter 將會把2中的handler包裝爲適配器,從而支持多種類型的處理器,即適配器設計模式的應用,從而很容易支持很多類型的處理器。DispatcherServlet中的HandlerAdapter列表如圖中所列的3種,依次遍歷,調用HanderAdapter.supports
判斷是否支持。這部分的詳細分析見【深入淺出spring】Spring MVC 流程解析 – HandlerAdapter
4. 調用Controller的具體方法處理請求,並返回一個 ModelAndView。HandlerAdapter會爲每一個請求生成一個ServletInvocableHandlerMethod
實例,核心方法invokeAndHandle
,包括輸入參數的處理和返回數據的解析。這部分的詳細分析見【深入淺出spring】Spring MVC 流程解析 – InvocableHandlerMethod
5. 視圖解析,遍歷DispatcherServlet
的ViewResolver列表,獲取對應的View對象,入口方法DispatcherServlet.processDispatchResult
6. 渲染,調用5中獲取的View的render方法,完成對Model數據的渲染。此處的 Model 實際是一個 Map 數據結構。
7. DispatcherServlet 將6中渲染後的數據返回響應給用戶,到此一個流程結束。
處理流程
附上代碼級的方法流,其中灰色標識主流程,綠色爲DispathcerServlet.doDispatch
方法中的流程,紅色爲HandlerAdapter.handler
方法中的流程
實現類羅列
handlerMappings:
- RequestMappingHandlerMapping
- BeanNameUrlHandlerMapping
- SimpleUrlHandlerMapping
- WelcomePageHandlerMapping
handlerAdapters:
- RequestMappingHandlerAdapter
- HttpRequestHandlerAdapter
- SimpleControllerHandlerAdapter
viewResolvers:
- ContentNegotiatingViewResolver
- BeanNameViewResolver
- ViewResolverComposite
- InternalResourceViewResolver