SpringMvc 處理請求的流程

時間全部的相遇,都是久別重逢---《一代宗師》java

前言

大半個月沒有寫博客了,墮落了。最近忽然想寫寫後端相關的博客了json

java 相關的後端框架接觸較深的有 Jfinal 和 SpringBoot。Jfinal 由於是國產的框架,當時內心也是有股執念在,就大體過了一遍源碼,後續的項目中只要是沒有技術棧限制,通常我都會用 Jfinal ,但漸漸發現生態相比 SpringBoot 仍是有差距的,爲了縮短項目開發週期,繼而轉變戰略去研究了一下 SpringBoot 。生態真的很重要啊!這真不是技術的高低問題。後端

沒接觸 SpringBoot 的時候,其實 Spring 玩的還行,可是仍是想折騰折騰。瞭解,熟練,精通,不看看源碼怎麼能算精通呢😆😆😆restful

一個請求怎麼從 Tomcat 轉發到 SpringBoot 中呢,一個請求又是怎麼在 SpringBoot 中處理的呢?翻翻源碼吧,不難,可是瞭解了對你想擴展和開發用處很大。比較喜歡用 SpringBoot 自帶的工具類 。app

源碼只大體講下邏輯,畢竟不是寫框架,不必逐字逐句的看框架

請求從 Tomcat 怎麼到 SpringBoot 呢

Tomcat 會根據請求找到對應的 Servlet ,而後根據 Servlet 和 Request 生成對應 FilterChain ,而後責任鏈模式執行 Filter 和 Servlet 達處處理請求的邏輯。工具

StandardWrapperValve.invoke(Request request, Response response)

// 分配servlet實例以處理此請求
if (!unavailable) {
    servlet = wrapper.allocate();
}
複製代碼
  • 代碼說明
@Slf4j
@WebServlet(name = "MyServletJson", urlPatterns = "/servlet/json", description = "Servlet的說明")
@Component
public class MyServletJson extends HttpServlet {
    
}
複製代碼

請求 url 爲 /servlet/json 時, servlet = wrapper.allocate(); 分配的爲 MyServletJson 實例post

url 沒有精準匹配到註冊的 Servlet , servlet = wrapper.allocate();分配的爲 DispatcherServlet 實例url

建立 FilterChain

// 爲當前請求建立 過濾器鏈
ApplicationFilterChain filterChain =
    ApplicationFilterFactory.createFilterChain(request, wrapper, servlet);
複製代碼

ApplicationFilterChain.doFilter 執行過濾器鏈

執行這個方法,會責任鏈模式執行過濾器和對應的 Servletspa

filterChain.doFilter(request.getRequest(), response.getResponse());
複製代碼

當執行到最後一個過濾器鏈的時候執行 Servlet.service 方法

ApplicationFilterChain.doFilter(ServletRequest request, ServletResponse response){
    internalDoFilter(request,response){
        servlet.service(request, response);
    }
}
複製代碼

以上代碼流程簡單展現了請求怎麼從 Tomcat 到 SpringBoot 框架中,簡單看下代碼,瞭解流程便可,咱們的重心應該放到 SpringBoot 中。

SpringBoot 怎麼樣處理請求呢

Tomcat 中 servlet.service(request, response); 將請求轉發到了 Servlet(DispatcherServlet) 中

Idea 中查看繼承關係,咱們主要關心的是 HttpServlet ,FrameworkServlet,DispatcherServlet。

debug 便可看到代碼走向,最終的處理邏輯集中在 DispatcherServlet.doDispatch 中

DispatcherServlet.doDispatch

protected void doDispatch(HttpServletRequest request, HttpServletResponse response) throws Exception {
    
    // 肯定當前請求的處理器執行鏈(肯定當前 request 執行那些攔截器)
    HandlerExecutionChain mappedHandler=getHandler(processedRequest);
    
    // 根據 handler 肯定當前請求的處理器適配器。
    HandlerAdapter ha = getHandlerAdapter(mappedHandler.getHandler());
    
    // 執行攔截器中的 preHandle 方法,責任鏈執行,有一個執行返回 false ,執行對應的 afterHandle 鏈結束
	if (!mappedHandler.applyPreHandle(processedRequest, response)) {
        return;
    }

    // 執行 aop 和 controller,service,dao, 
    // restful接口時,在這個邏輯經過jackson 將數據寫入到 response而且返回的 mv=null;
    mv = ha.handle(processedRequest, response, mappedHandler.getHandler());

    // 設置 ModelAndView 中的默認 View
    applyDefaultViewName(processedRequest, mv);
    
    // 鏈式執行攔截器中 postHandle
    mappedHandler.applyPostHandle(processedRequest, response, mv);
    
    // View.render 將數據寫入到 Response 中返回
    processDispatchResult(processedRequest, response, mappedHandler, mv, dispatchException);
    
    // 鏈式執行攔截器中 afterHandle
    mappedHandler.applyAfterConcurrentHandlingStarted(processedRequest, response);
}
複製代碼

圖也不行畫了,直接網上找了一個😂😂😂

視圖渲染部分(View.render),咱們也能夠封裝本身 View。好比經常使用的文件下載,咱們能夠封裝本身 View ,只須要傳入對應文件流和文件名稱便可達到下載的目的

根據 HandlerMapping 獲取 HandlerExecutionChain

HandlerExecutionChain mappedHandler=getHandler(processedRequest);

經常使用的 HandlerMapping有:

  • RequestMappingHandlerMapping

處理 restful 接口,對應的handler 爲 HandlerMethod

使用 RequestMappingHandlerMapping 咱們能夠拿到 url 對應的 HandlerMethod,HandlerMethod 能夠得到方法和類上的註解能夠進行基礎接口權限數據的初始化,今後告別手動補數據

  • SimpleUrlHandlerMapping

轉發或重定向連接,在此配置的WebMvcConfigurer.addViewControllers();handler 爲ParameterizableViewController

  • SimpleUrlHandlerMapping 用於處理靜態資源

處理頁面或者靜態資源,handler 爲 ResourceHttpRequestHandler

根據處理器獲取處理器適配器

HandlerAdapter ha = getHandlerAdapter(mappedHandler.getHandler());

經常使用的 HandlerAdapter 有:

  • RequestMappingHandlerAdapter

處理 restful 接口,handler 爲 HandlerMethod

  • SimpleControllerHandlerAdapter

處理簡單的 url 轉發或重定向,WebMvcConfigurer.addViewControllers();handler 爲ParameterizableViewController

  • HttpRequestHandlerAdapter

處理靜態資源,handler 爲 ResourceHttpRequestHandler

相關文章
相關標籤/搜索