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

相关文章
相关标签/搜索