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

- 用戶發(fā)起請求,到
DispatcherServlet; - 然后到
HandlerMapping返回處理器鏈(包含攔截器和具體處理的Handler); - 調(diào)用處理器鏈的適配器 HandlerAdapter 來處理;
- 執(zhí)行具體的方法,比如
@RequestMapper修飾的邏輯處理方法; - 返回結(jié)果的視圖解析器;
- 最后進(jìn)行視圖解析和渲染返回結(jié)果給用戶;
DispatcherServlet
DispatcherServlet是前置控制器,配置在web.xml文件中的。攔截匹配的請求,Servlet攔截匹配規(guī)則要自己定義,把攔截下來的請求,依據(jù)相應(yīng)的規(guī)則分發(fā)到目標(biāo)Controller來處理,是配置spring MVC的第一步。 DispatcherServlet是前端控制器設(shè)計模式的實現(xiàn),提供Spring Web MVC的集中訪問點,而且負(fù)責(zé)職責(zé)的分派,而且與Spring IoC容器無縫集成,從而可以獲得Spring的所有好處。
源碼分析
org.springframework.web.servlet.DispatcherServlet#doDispatch 方法是主要處理請求的源碼如下:
protected void doDispatch(HttpServletRequest request, HttpServletResponse response) throws Exception {
try {
try {
// 文件上傳相關(guān)
processedRequest = checkMultipart(request);
multipartRequestParsed = (processedRequest != request);
// DispatcherServlet收到請求調(diào)用處理器映射器HandlerMapping。
// 處理器映射器根據(jù)請求url找到具體的處理器,生成處理器執(zhí)行鏈HandlerExecutionChain(包括處理器對象和處理器攔截器)一并返回給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ù)驗證等操作
// 執(zhí)行處理器Handler(Controller,也叫頁面控制器)。
// Handler執(zhí)行完成返回ModelAndView
// HandlerAdapter將Handler執(zhí)行結(jié)果ModelAndView返回到DispatcherServlet
mv = ha.handle(processedRequest, response, mappedHandler.getHandler());
if (asyncManager.isConcurrentHandlingStarted()) {
return;
}
// 如果沒有視圖,給你設(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對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、文件上傳解析,如果請求類型是multipart將通過MultipartResolver進(jìn)行文件上傳解析; 2、通過HandlerMapping,將請求映射到處理器(返回一個HandlerExecutionChain,它包括一個處理器、多個HandlerInterceptor攔截器); 3、通過HandlerAdapter支持多種類型的處理器(HandlerExecutionChain中的處理器); 4、通過ViewResolver解析邏輯視圖名到具體視圖實現(xiàn); 5、本地化解析; 6、渲染具體的視圖等; 7、如果執(zhí)行過程中遇到異常將交給HandlerExceptionResolver來解析。
DispatcherServlet初始化的上下文加載的Bean是只對SpringMVC有效的Bean, 如Controller、HandlerMapping、HandlerAdapter等等,該初始化上下文只加載Web相關(guān)組件。
DispatcherServlet初始化主要做了如下兩件事情: 1、初始化SpringMVC使用的Web上下文,并且可能指定父容器為(ContextLoaderListener加載了根上下文); 2、初始化DispatcherServlet使用的策略,如HandlerMapping、HandlerAdapter等。
Spring MVC 中的一些核心類
DispatcherServlet 默認(rèn)使用 WebApplicationContext 作為上下文,該上下文中特殊的Bean有一下幾個:
| 類名 | 描述 |
|---|---|
| Controller | 處理器/頁面控制器,做的是MVC中的C的事情,但控制邏輯轉(zhuǎn)移到前端控制器了,用于對請求進(jìn)行處理; |
| HandlerMapping | 請求到處理器的映射,如果映射成功返回一個HandlerExecutionChain(包含一個Handler處理器(頁面控制器)對象、多個HandlerInterceptor攔截器)對象;如BeanNameUrlHandlerMapping將URL與Bean名字映射,映射成功的Bean就是此處的處理器; |
| HandlerMapping | 請求到處理器的映射,如果映射成功返回一個HandlerExecutionChain對象(包含一個Handler處理器(頁面控制器)對象、多個HandlerInterceptor攔截器)對象;如BeanNameUrlHandlerMapping將URL與Bean名字映射,映射成功的Bean就是此處的處理器; |
| ViewResolver | ViewResolver將把邏輯視圖名解析為具體的View,通過這種策略模式,很容易更換其他視圖技術(shù);如InternalResourceViewResolver將邏輯視圖名映射為jsp視圖; |
| LocalResover | 本地化解析,因為Spring支持國際化,因此LocalResover解析客戶端的Locale信息從而方便進(jìn)行國際化; |
| ThemeResovler | 主題解析,通過它來實現(xiàn)一個頁面多套風(fēng)格,即常見的類似于軟件皮膚效果; |
| MultipartResolver | 文件上傳解析,用于支持文件上傳; |
| HandlerExceptionResolver | 處理器異常解析,可以將異常映射到相應(yīng)的統(tǒng)一錯誤界面,從而顯示用戶友好的界面(而不是給用戶看到具體的錯誤信息); |
| RequestToViewNameTranslator | 當(dāng)處理器沒有返回邏輯視圖名等相關(guān)信息時,自動將請求URL映射為邏輯視圖名; |
到此這篇關(guān)于Spring MVC 前端控制器 (DispatcherServlet)處理流程的文章就介紹到這了,更多相關(guān)Spring MVC 處理流程內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
Spring Boot 非web應(yīng)用程序的實現(xiàn)
SpringBoot框架中,要創(chuàng)建一個非Web應(yīng)用程序(純 Java 程序),有兩種方式,下面就來介紹一下,感興趣的可以來了解一下2025-03-03
Java?ConcurrentHashMap實現(xiàn)線程安全的代碼示例
眾所周知ConcurrentHashMap是HashMap的多線程版本,HashMap?在并發(fā)操作時會有各種問題,而這些問題,只要使用ConcurrentHashMap就可以完美解決了,本文將給詳細(xì)介紹ConcurrentHashMap是如何保證線程安全的2023-05-05
SpringBoot靜態(tài)視頻實時播放的實現(xiàn)代碼
這篇文章主要介紹了SpringBoot靜態(tài)視頻實時播放的實現(xiàn)代碼,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2020-01-01
JAVA多線程之實現(xiàn)用戶任務(wù)排隊并預(yù)估排隊時長
本文主要介紹了Java多線程之實現(xiàn)用戶任務(wù)排隊并預(yù)估排隊時長的問題,文中的代碼具有一定的學(xué)習(xí)和工作價值,感興趣的小伙伴快跟隨小編一起學(xué)習(xí)一下吧2021-12-12

