Spring?MVC?前端控制器?(DispatcherServlet)處理流程解析
Spring MVC 請(qǐng)求處理流程
- 用戶(hù)發(fā)起請(qǐng)求,到
DispatcherServlet
; - 然后到
HandlerMapping
返回處理器鏈(包含攔截器和具體處理的Handler
); - 調(diào)用處理器鏈的適配器 HandlerAdapter 來(lái)處理;
- 執(zhí)行具體的方法,比如
@RequestMapper
修飾的邏輯處理方法; - 返回結(jié)果的視圖解析器;
- 最后進(jìn)行視圖解析和渲染返回結(jié)果給用戶(hù);
DispatcherServlet
DispatcherServlet是前置控制器,配置在web.xml文件中的。攔截匹配的請(qǐng)求,Servlet攔截匹配規(guī)則要自己定義,把攔截下來(lái)的請(qǐng)求,依據(jù)相應(yīng)的規(guī)則分發(fā)到目標(biāo)Controller來(lái)處理,是配置spring MVC的第一步。 DispatcherServlet是前端控制器設(shè)計(jì)模式的實(shí)現(xiàn),提供Spring Web MVC的集中訪問(wèn)點(diǎn),而且負(fù)責(zé)職責(zé)的分派,而且與Spring IoC容器無(wú)縫集成,從而可以獲得Spring的所有好處。
源碼分析
org.springframework.web.servlet.DispatcherServlet#doDispatch
方法是主要處理請(qǐng)求的源碼如下:
protected void doDispatch(HttpServletRequest request, HttpServletResponse response) throws Exception { try { try { // 文件上傳相關(guān) processedRequest = checkMultipart(request); multipartRequestParsed = (processedRequest != request); // DispatcherServlet收到請(qǐng)求調(diào)用處理器映射器HandlerMapping。 // 處理器映射器根據(jù)請(qǐng)求url找到具體的處理器,生成處理器執(zhí)行鏈HandlerExecutionChain(包括處理器對(duì)象和處理器攔截器)一并返回給DispatcherServlet。 mappedHandler = getHandler(processedRequest); if (mappedHandler == null) { noHandlerFound(processedRequest, response); return; } 4.DispatcherServlet根據(jù)處理器Handler獲取處理器適配器HandlerAdapter, HandlerAdapter ha = getHandlerAdapter(mappedHandler.getHandler()); // Process last-modified header, if supported by the handler. HTTP緩存相關(guān) String method = request.getMethod(); boolean isGet = HttpMethod.GET.matches(method); if (isGet || HttpMethod.HEAD.matches(method)) { long lastModified = ha.getLastModified(request, mappedHandler.getHandler()); if (new ServletWebRequest(request, response).checkNotModified(lastModified) && isGet) { return; } } // 前置攔截器 if (!mappedHandler.applyPreHandle(processedRequest, response)) { // 返回false就不進(jìn)行后續(xù)處理了 return; } // 執(zhí)行HandlerAdapter處理一系列的操作,如:參數(shù)封裝,數(shù)據(jù)格式轉(zhuǎn)換,數(shù)據(jù)驗(yàn)證等操作 // 執(zhí)行處理器Handler(Controller,也叫頁(yè)面控制器)。 // Handler執(zhí)行完成返回ModelAndView // HandlerAdapter將Handler執(zhí)行結(jié)果ModelAndView返回到DispatcherServlet mv = ha.handle(processedRequest, response, mappedHandler.getHandler()); if (asyncManager.isConcurrentHandlingStarted()) { return; } // 如果沒(méi)有視圖,給你設(shè)置默認(rèn)視圖 json忽略 applyDefaultViewName(processedRequest, mv); //后置攔截器 mappedHandler.applyPostHandle(processedRequest, response, mv); } catch (Exception ex) { dispatchException = ex; } catch (Throwable err) { // As of 4.3, we're processing Errors thrown from handler methods as well, // making them available for @ExceptionHandler methods and other scenarios. dispatchException = new NestedServletException("Handler dispatch failed", err); } // DispatcherServlet將ModelAndView傳給ViewReslover視圖解析器 // ViewReslover解析后返回具體View // DispatcherServlet對(duì)View進(jìn)行渲染視圖(即將模型數(shù)據(jù)model填充至視圖中)。 // DispatcherServlet響應(yīng)用戶(hù)。 processDispatchResult(processedRequest, response, mappedHandler, mv, dispatchException); } catch (Exception ex) { triggerAfterCompletion(processedRequest, response, mappedHandler, ex); } catch (Throwable err) { triggerAfterCompletion(processedRequest, response, mappedHandler, new NestedServletException("Handler processing failed", err)); } finally { if (asyncManager.isConcurrentHandlingStarted()) { // Instead of postHandle and afterCompletion if (mappedHandler != null) { mappedHandler.applyAfterConcurrentHandlingStarted(processedRequest, response); } } else { // Clean up any resources used by a multipart request. if (multipartRequestParsed) { cleanupMultipart(processedRequest); } } } }
在doDispatch
方中已經(jīng)涵蓋了DispatcherServlet
的主要職責(zé): 1、文件上傳解析,如果請(qǐng)求類(lèi)型是multipart將通過(guò)MultipartResolver進(jìn)行文件上傳解析; 2、通過(guò)HandlerMapping,將請(qǐng)求映射到處理器(返回一個(gè)HandlerExecutionChain,它包括一個(gè)處理器、多個(gè)HandlerInterceptor攔截器); 3、通過(guò)HandlerAdapter支持多種類(lèi)型的處理器(HandlerExecutionChain中的處理器); 4、通過(guò)ViewResolver解析邏輯視圖名到具體視圖實(shí)現(xiàn); 5、本地化解析; 6、渲染具體的視圖等; 7、如果執(zhí)行過(guò)程中遇到異常將交給HandlerExceptionResolver來(lái)解析。
DispatcherServlet初始化的上下文加載的Bean是只對(duì)SpringMVC有效的Bean, 如Controller、HandlerMapping、HandlerAdapter等等,該初始化上下文只加載Web相關(guān)組件。
DispatcherServlet初始化主要做了如下兩件事情: 1、初始化SpringMVC使用的Web上下文,并且可能指定父容器為(ContextLoaderListener加載了根上下文); 2、初始化DispatcherServlet使用的策略,如HandlerMapping、HandlerAdapter等。
Spring MVC 中的一些核心類(lèi)
DispatcherServlet 默認(rèn)使用 WebApplicationContext 作為上下文,該上下文中特殊的Bean有一下幾個(gè):
類(lèi)名 | 描述 |
---|---|
Controller | 處理器/頁(yè)面控制器,做的是MVC中的C的事情,但控制邏輯轉(zhuǎn)移到前端控制器了,用于對(duì)請(qǐng)求進(jìn)行處理; |
HandlerMapping | 請(qǐng)求到處理器的映射,如果映射成功返回一個(gè)HandlerExecutionChain(包含一個(gè)Handler處理器(頁(yè)面控制器)對(duì)象、多個(gè)HandlerInterceptor攔截器)對(duì)象;如BeanNameUrlHandlerMapping將URL與Bean名字映射,映射成功的Bean就是此處的處理器; |
HandlerMapping | 請(qǐng)求到處理器的映射,如果映射成功返回一個(gè)HandlerExecutionChain對(duì)象(包含一個(gè)Handler處理器(頁(yè)面控制器)對(duì)象、多個(gè)HandlerInterceptor攔截器)對(duì)象;如BeanNameUrlHandlerMapping將URL與Bean名字映射,映射成功的Bean就是此處的處理器; |
ViewResolver | ViewResolver將把邏輯視圖名解析為具體的View,通過(guò)這種策略模式,很容易更換其他視圖技術(shù);如InternalResourceViewResolver將邏輯視圖名映射為jsp視圖; |
LocalResover | 本地化解析,因?yàn)镾pring支持國(guó)際化,因此LocalResover解析客戶(hù)端的Locale信息從而方便進(jìn)行國(guó)際化; |
ThemeResovler | 主題解析,通過(guò)它來(lái)實(shí)現(xiàn)一個(gè)頁(yè)面多套風(fēng)格,即常見(jiàn)的類(lèi)似于軟件皮膚效果; |
MultipartResolver | 文件上傳解析,用于支持文件上傳; |
HandlerExceptionResolver | 處理器異常解析,可以將異常映射到相應(yīng)的統(tǒng)一錯(cuò)誤界面,從而顯示用戶(hù)友好的界面(而不是給用戶(hù)看到具體的錯(cuò)誤信息); |
RequestToViewNameTranslator | 當(dāng)處理器沒(méi)有返回邏輯視圖名等相關(guān)信息時(shí),自動(dòng)將請(qǐng)求URL映射為邏輯視圖名; |
到此這篇關(guān)于Spring MVC 前端控制器 (DispatcherServlet)處理流程的文章就介紹到這了,更多相關(guān)Spring MVC 處理流程內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
java使用common-httpclient包實(shí)現(xiàn)post請(qǐng)求方法示例
這篇文章主要給大家介紹了關(guān)于java使用common-httpclient包實(shí)現(xiàn)post請(qǐng)求的相關(guān)資料,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2018-08-08一個(gè)例子帶你看懂Java中synchronized關(guān)鍵字到底怎么用
synchronized是Java里的一個(gè)關(guān)鍵字,起到的一個(gè)效果是"監(jiān)視器鎖",它的功能就是保證操作的原子性,同時(shí)禁止指令重排序和保證內(nèi)存的可見(jiàn)性,下面這篇文章主要給大家介紹了關(guān)于如何通過(guò)一個(gè)例子帶你看懂Java中synchronized關(guān)鍵字到底怎么用的相關(guān)資料,需要的朋友可以參考下2022-10-10基于java計(jì)算買(mǎi)賣(mài)股票的最佳時(shí)機(jī)
這篇文章主要介紹了基于java計(jì)算買(mǎi)賣(mài)股票的最佳時(shí)機(jī),文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下2019-10-10Springboot啟動(dòng)停止命令的.sh腳本編寫(xiě)方式
這篇文章主要介紹了Springboot啟動(dòng)停止命令的.sh腳本編寫(xiě)方式,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2023-05-05FastJson踩坑:@JsonField在反序列化時(shí)失效的解決
這篇文章主要介紹了FastJson踩坑:@JsonField在反序列化時(shí)失效的解決方案,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2021-06-06SpringBoot+JUnit5+MockMvc+Mockito單元測(cè)試的實(shí)現(xiàn)
今天聊聊如何在 SpringBoot 中集成 Junit5、MockMvc、Mocktio。Junit5 是在 Java 棧中應(yīng)用最廣的測(cè)試框架,文中通過(guò)示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2021-09-09實(shí)戰(zhàn)分布式醫(yī)療掛號(hào)通用模塊統(tǒng)一返回結(jié)果異常日志處理
這篇文章主要為大家介紹了實(shí)戰(zhàn)分布式醫(yī)療掛號(hào)系統(tǒng)之統(tǒng)一返回結(jié)果統(tǒng)一異常處理,統(tǒng)一日志處理到通用模塊示例詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助2022-04-04