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

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

 更新時(shí)間:2022年05月28日 09:47:07   作者:心城以北  
DispatcherServlet是前置控制器,配置在web.xml文件中的,這篇文章主要介紹了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就是此處的處理器;
ViewResolverViewResolver將把邏輯視圖名解析為具體的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)文章

  • 老生常談 MyBatis 復(fù)雜查詢(xún)

    老生常談 MyBatis 復(fù)雜查詢(xún)

    這篇文章主要介紹了 MyBatis 復(fù)雜查詢(xún)的相關(guān)知識(shí),本文給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下
    2021-01-01
  • java使用common-httpclient包實(shí)現(xiàn)post請(qǐng)求方法示例

    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)鍵字到底怎么用

    一個(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ī)

    這篇文章主要介紹了基于java計(jì)算買(mǎi)賣(mài)股票的最佳時(shí)機(jī),文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下
    2019-10-10
  • Springboot啟動(dòng)停止命令的.sh腳本編寫(xiě)方式

    Springboot啟動(dòng)停止命令的.sh腳本編寫(xiě)方式

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

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

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

    SpringBoot+JUnit5+MockMvc+Mockito單元測(cè)試的實(shí)現(xiàn)

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

    通過(guò)java生成讀取二維碼詳解

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

    實(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
  • Java對(duì)象初始化順序的使用

    Java對(duì)象初始化順序的使用

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

最新評(píng)論