欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

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

 更新時間:2022年05月28日 09:47:07   作者:心城以北  
DispatcherServlet是前置控制器,配置在web.xml文件中的,這篇文章主要介紹了Spring?MVC?前端控制器?(DispatcherServlet)處理流程,需要的朋友可以參考下

Spring MVC 請求處理流程

  • 用戶發(fā)起請求,到 DispatcherServlet;
  • 然后到 HandlerMapping 返回處理器鏈(包含攔截器和具體處理的 Handler);
  • 調用處理器鏈的適配器 HandlerAdapter 來處理;
  • 執(zhí)行具體的方法,比如 @RequestMapper修飾的邏輯處理方法;
  • 返回結果的視圖解析器;
  • 最后進行視圖解析和渲染返回結果給用戶;

DispatcherServlet

DispatcherServlet是前置控制器,配置在web.xml文件中的。攔截匹配的請求,Servlet攔截匹配規(guī)則要自己定義,把攔截下來的請求,依據相應的規(guī)則分發(fā)到目標Controller來處理,是配置spring MVC的第一步。 DispatcherServlet是前端控制器設計模式的實現,提供Spring Web MVC的集中訪問點,而且負責職責的分派,而且與Spring IoC容器無縫集成,從而可以獲得Spring的所有好處。

源碼分析

org.springframework.web.servlet.DispatcherServlet#doDispatch 方法是主要處理請求的源碼如下:

protected void doDispatch(HttpServletRequest request, HttpServletResponse response) throws Exception {
    try {
        try {
            // 文件上傳相關
            processedRequest = checkMultipart(request);
            multipartRequestParsed = (processedRequest != request);
            // DispatcherServlet收到請求調用處理器映射器HandlerMapping。
            // 處理器映射器根據請求url找到具體的處理器,生成處理器執(zhí)行鏈HandlerExecutionChain(包括處理器對象和處理器攔截器)一并返回給DispatcherServlet。
            mappedHandler = getHandler(processedRequest);
            if (mappedHandler == null) {
                noHandlerFound(processedRequest, response);
                return;
            }
            4.DispatcherServlet根據處理器Handler獲取處理器適配器HandlerAdapter,
            HandlerAdapter ha = getHandlerAdapter(mappedHandler.getHandler());
            // Process last-modified header, if supported by the handler.  HTTP緩存相關
            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就不進行后續(xù)處理了
                return;
            }
            // 執(zhí)行HandlerAdapter處理一系列的操作,如:參數封裝,數據格式轉換,數據驗證等操作
            // 執(zhí)行處理器Handler(Controller,也叫頁面控制器)。
            // Handler執(zhí)行完成返回ModelAndView
            // HandlerAdapter將Handler執(zhí)行結果ModelAndView返回到DispatcherServlet
            mv = ha.handle(processedRequest, response, mappedHandler.getHandler());
            if (asyncManager.isConcurrentHandlingStarted()) {
                return;
            }
            // 如果沒有視圖,給你設置默認視圖  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進行渲染視圖(即將模型數據model填充至視圖中)。
        // DispatcherServlet響應用戶。
        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方中已經涵蓋了DispatcherServlet的主要職責: 1、文件上傳解析,如果請求類型是multipart將通過MultipartResolver進行文件上傳解析; 2、通過HandlerMapping,將請求映射到處理器(返回一個HandlerExecutionChain,它包括一個處理器、多個HandlerInterceptor攔截器); 3、通過HandlerAdapter支持多種類型的處理器(HandlerExecutionChain中的處理器); 4、通過ViewResolver解析邏輯視圖名到具體視圖實現; 5、本地化解析; 6、渲染具體的視圖等; 7、如果執(zhí)行過程中遇到異常將交給HandlerExceptionResolver來解析。

DispatcherServlet初始化的上下文加載的Bean是只對SpringMVC有效的Bean, 如Controller、HandlerMapping、HandlerAdapter等等,該初始化上下文只加載Web相關組件。

DispatcherServlet初始化主要做了如下兩件事情: 1、初始化SpringMVC使用的Web上下文,并且可能指定父容器為(ContextLoaderListener加載了根上下文); 2、初始化DispatcherServlet使用的策略,如HandlerMapping、HandlerAdapter等。

Spring MVC 中的一些核心類

DispatcherServlet 默認使用 WebApplicationContext 作為上下文,該上下文中特殊的Bean有一下幾個:

類名描述
Controller處理器/頁面控制器,做的是MVC中的C的事情,但控制邏輯轉移到前端控制器了,用于對請求進行處理;
HandlerMapping請求到處理器的映射,如果映射成功返回一個HandlerExecutionChain(包含一個Handler處理器(頁面控制器)對象、多個HandlerInterceptor攔截器)對象;如BeanNameUrlHandlerMapping將URL與Bean名字映射,映射成功的Bean就是此處的處理器;
HandlerMapping請求到處理器的映射,如果映射成功返回一個HandlerExecutionChain對象(包含一個Handler處理器(頁面控制器)對象、多個HandlerInterceptor攔截器)對象;如BeanNameUrlHandlerMapping將URL與Bean名字映射,映射成功的Bean就是此處的處理器;
ViewResolverViewResolver將把邏輯視圖名解析為具體的View,通過這種策略模式,很容易更換其他視圖技術;如InternalResourceViewResolver將邏輯視圖名映射為jsp視圖;
LocalResover本地化解析,因為Spring支持國際化,因此LocalResover解析客戶端的Locale信息從而方便進行國際化;
ThemeResovler主題解析,通過它來實現一個頁面多套風格,即常見的類似于軟件皮膚效果;
MultipartResolver文件上傳解析,用于支持文件上傳;
HandlerExceptionResolver處理器異常解析,可以將異常映射到相應的統(tǒng)一錯誤界面,從而顯示用戶友好的界面(而不是給用戶看到具體的錯誤信息);
RequestToViewNameTranslator當處理器沒有返回邏輯視圖名等相關信息時,自動將請求URL映射為邏輯視圖名;

到此這篇關于Spring MVC 前端控制器 (DispatcherServlet)處理流程的文章就介紹到這了,更多相關Spring MVC 處理流程內容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關文章希望大家以后多多支持腳本之家!

相關文章

  • 老生常談 MyBatis 復雜查詢

    老生常談 MyBatis 復雜查詢

    這篇文章主要介紹了 MyBatis 復雜查詢的相關知識,本文給大家介紹的非常詳細,對大家的學習或工作具有一定的參考借鑒價值,需要的朋友可以參考下
    2021-01-01
  • java使用common-httpclient包實現post請求方法示例

    java使用common-httpclient包實現post請求方法示例

    這篇文章主要給大家介紹了關于java使用common-httpclient包實現post請求的相關資料,文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友們下面隨著小編來一起學習學習吧
    2018-08-08
  • 一個例子帶你看懂Java中synchronized關鍵字到底怎么用

    一個例子帶你看懂Java中synchronized關鍵字到底怎么用

    synchronized是Java里的一個關鍵字,起到的一個效果是"監(jiān)視器鎖",它的功能就是保證操作的原子性,同時禁止指令重排序和保證內存的可見性,下面這篇文章主要給大家介紹了關于如何通過一個例子帶你看懂Java中synchronized關鍵字到底怎么用的相關資料,需要的朋友可以參考下
    2022-10-10
  • 基于java計算買賣股票的最佳時機

    基于java計算買賣股票的最佳時機

    這篇文章主要介紹了基于java計算買賣股票的最佳時機,文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友可以參考下
    2019-10-10
  • Springboot啟動停止命令的.sh腳本編寫方式

    Springboot啟動停止命令的.sh腳本編寫方式

    這篇文章主要介紹了Springboot啟動停止命令的.sh腳本編寫方式,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教
    2023-05-05
  • FastJson踩坑:@JsonField在反序列化時失效的解決

    FastJson踩坑:@JsonField在反序列化時失效的解決

    這篇文章主要介紹了FastJson踩坑:@JsonField在反序列化時失效的解決方案,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教
    2021-06-06
  • SpringBoot+JUnit5+MockMvc+Mockito單元測試的實現

    SpringBoot+JUnit5+MockMvc+Mockito單元測試的實現

    今天聊聊如何在 SpringBoot 中集成 Junit5、MockMvc、Mocktio。Junit5 是在 Java 棧中應用最廣的測試框架,文中通過示例代碼介紹的非常詳細,具有一定的參考價值,感興趣的小伙伴們可以參考一下
    2021-09-09
  • 通過java生成讀取二維碼詳解

    通過java生成讀取二維碼詳解

    這篇文章主要介紹了java二維碼生成讀取詳解,二維碼再生活在無處不在,文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,下面和小編一起來學習一下吧
    2019-05-05
  • 實戰(zhàn)分布式醫(yī)療掛號通用模塊統(tǒng)一返回結果異常日志處理

    實戰(zhàn)分布式醫(yī)療掛號通用模塊統(tǒng)一返回結果異常日志處理

    這篇文章主要為大家介紹了實戰(zhàn)分布式醫(yī)療掛號系統(tǒng)之統(tǒng)一返回結果統(tǒng)一異常處理,統(tǒng)一日志處理到通用模塊示例詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助
    2022-04-04
  • Java對象初始化順序的使用

    Java對象初始化順序的使用

    本篇文章介紹了,Java對象初始化順序的使用。需要的朋友參考下
    2013-04-04

最新評論