Spring?MVC?前端控制器?(DispatcherServlet)處理流程解析
Spring MVC 請(qǐng)求處理流程

- 用戶發(fā)起請(qǐng)求,到
DispatcherServlet; - 然后到
HandlerMapping返回處理器鏈(包含攔截器和具體處理的Handler); - 調(diào)用處理器鏈的適配器 HandlerAdapter 來(lái)處理;
- 執(zhí)行具體的方法,比如
@RequestMapper修飾的邏輯處理方法; - 返回結(jié)果的視圖解析器;
- 最后進(jìn)行視圖解析和渲染返回結(jié)果給用戶;
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)用戶。
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)求類型是multipart將通過(guò)MultipartResolver進(jìn)行文件上傳解析; 2、通過(guò)HandlerMapping,將請(qǐng)求映射到處理器(返回一個(gè)HandlerExecutionChain,它包括一個(gè)處理器、多個(gè)HandlerInterceptor攔截器); 3、通過(guò)HandlerAdapter支持多種類型的處理器(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 中的一些核心類
DispatcherServlet 默認(rèn)使用 WebApplicationContext 作為上下文,該上下文中特殊的Bean有一下幾個(gè):
| 類名 | 描述 |
|---|---|
| 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解析客戶端的Locale信息從而方便進(jìn)行國(guó)際化; |
| ThemeResovler | 主題解析,通過(guò)它來(lái)實(shí)現(xiàn)一個(gè)頁(yè)面多套風(fēng)格,即常見的類似于軟件皮膚效果; |
| MultipartResolver | 文件上傳解析,用于支持文件上傳; |
| HandlerExceptionResolver | 處理器異常解析,可以將異常映射到相應(yīng)的統(tǒng)一錯(cuò)誤界面,從而顯示用戶友好的界面(而不是給用戶看到具體的錯(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)存的可見性,下面這篇文章主要給大家介紹了關(guān)于如何通過(guò)一個(gè)例子帶你看懂Java中synchronized關(guān)鍵字到底怎么用的相關(guān)資料,需要的朋友可以參考下2022-10-10
基于java計(jì)算買賣股票的最佳時(shí)機(jī)
這篇文章主要介紹了基于java計(jì)算買賣股票的最佳時(shí)機(jī),文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下2019-10-10
Springboot啟動(dòng)停止命令的.sh腳本編寫方式
這篇文章主要介紹了Springboot啟動(dòng)停止命令的.sh腳本編寫方式,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2023-05-05
FastJson踩坑:@JsonField在反序列化時(shí)失效的解決
這篇文章主要介紹了FastJson踩坑:@JsonField在反序列化時(shí)失效的解決方案,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2021-06-06
SpringBoot+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

