springMVC的执行流程

当浏览器发送一个请求 http://localhost:8080/hello 后,请求到达服务器,其处理流程是:

  1. 服务器提供了 DispatcherServlet,它使用的是标准 Servlet 技术

    • 路径:默认映射路径为 /,即会匹配到所有请求 URL,可作为请求的统一入口,也被称之为前控制器

      • jsp 不会匹配到 DispatcherServlet

      • 其它有路径的 Servlet 匹配优先级也高于 DispatcherServlet

    • 创建:在 Boot 中,由 DispatcherServletAutoConfiguration 这个自动配置类提供 DispatcherServlet 的 bean

    • 初始化:DispatcherServlet 初始化时会优先到容器里寻找各种组件,作为它的成员变量

      • HandlerMapping,初始化时记录映射关系

      • HandlerAdapter,初始化时准备参数解析器、返回值处理器、消息转换器

      • HandlerExceptionResolver,初始化时准备参数解析器、返回值处理器、消息转换器

      • ViewResolver

  2. DispatcherServlet 会利用 RequestMappingHandlerMapping 查找控制器方法

    • 例如根据 /hello 路径找到 @RequestMapping("/hello") 对应的控制器方法

    • 控制器方法会被封装为 HandlerMethod 对象,并结合匹配到的拦截器一起返回给 DispatcherServlet

    • HandlerMethod 和拦截器合在一起称为 HandlerExecutionChain(调用链)对象

  3. DispatcherServlet 接下来会:

    1. 调用拦截器的 preHandle 方法

    2. RequestMappingHandlerAdapter 调用 handle 方法,准备数据绑定工厂、模型工厂、ModelAndViewContainer、将 HandlerMethod 完善为 ServletInvocableHandlerMethod

      • @ControllerAdvice 全局增强点1️⃣:补充模型数据

      • @ControllerAdvice 全局增强点2️⃣:补充自定义类型转换器

      • 使用 HandlerMethodArgumentResolver 准备参数

        • @ControllerAdvice 全局增强点3️⃣:RequestBody 增强

      • 调用 ServletInvocableHandlerMethod

      • 使用 HandlerMethodReturnValueHandler 处理返回值

        • @ControllerAdvice 全局增强点4️⃣:ResponseBody 增强

      • 根据 ModelAndViewContainer 获取 ModelAndView

        • 如果返回的 ModelAndView 为 null,不走第 4 步视图解析及渲染流程

          • 例如,有的返回值处理器调用了 HttpMessageConverter 来将结果转换为 JSON,这时 ModelAndView 就为 null

        • 如果返回的 ModelAndView 不为 null,会在第 4 步走视图解析及渲染流程

    3. 调用拦截器的 postHandle 方法

    4. 处理异常或视图渲染

      • 如果 1~3 出现异常,走 ExceptionHandlerExceptionResolver 处理异常流程

        • @ControllerAdvice 全局增强点5️⃣:@ExceptionHandler 异常处理

      • 正常,走视图解析及渲染流程

    5. 调用拦截器的 afterCompletion 方法

初始化阶段

  1. 在 Web 容器第一次用到 DispatcherServlet 的时候,会创建其对象(在spring和springMVC整合的时候是由tomcat初始化并创建的,到了springboot的时候就是由spring创建的)并执行 init 方法

  2. init 方法内会创建 Spring Web 容器,并调用容器 refresh 方法

  3. refresh 过程中会创建并初始化 SpringMVC 中的重要组件, 例如 MultipartResolver(文件上传、表单处理),HandlerMapping(负责请求映射),HandlerAdapter(调用控制器的方法),HandlerExceptionResolver(处理异常)、ViewResolver (把字符串解析成视图对象)等

  4. 容器初始化后,会将上一步初始化好的重要组件,赋值给 DispatcherServlet 的成员变量,留待后用

     

匹配阶段 

  1. 用户发送的请求统一到达前端控制器 DispatcherServlet(虽然是所有请求的入口,但不处理具体请求)

  2. DispatcherServlet 遍历所有 HandlerMapping ,找到与路径匹配的处理器

    HandlerMapping 有多个,每个 HandlerMapping 会返回不同的处理器对象,谁先匹配,返回谁的处理器。其中能识别 @RequestMapping 的优先级最高

    ② 对应 @RequestMapping 的处理器是 HandlerMethod,它包含了控制器对象和控制器方法信息

    ③ 其中路径与处理器的映射关系在 HandlerMapping 初始化时就会建立好

     

  3. 遍历HandlerAdapter 处理器适配器,找到能处理 HandlerMethod 的适配器对象,开始调用

 

  4. 将 HandlerMethod 连同匹配到的拦截器,生成调用链对象 HandlerExecutionChain 返回

执行阶段

 1、执行拦截器 preHandle,返回值为true就放行,反之就拦截下来了

 

2、由 HandlerAdapter 调用 HandlerMethod

① 调用前处理不同类型的参数,参数解析好之后,通过method的反射调用把bean和参数传入。

② 调用后处理不同类型的返回值

 

3、第 2 步没有异常

① 返回 ModelAndView

② 执行拦截器 postHandle 方法,跟prehandle顺序反过来

③ 解析视图(字符串利用ViewResolver),得到 View 对象,进行视图渲染

 

4、第 2 步有异常,进入 HandlerExceptionResolver 异常处理流程

 

5、最后都会执行拦截器的 afterCompletion 方法

6、如果控制器方法标注了 @ResponseBody 注解,则在第 2 步,就会生成 json 结果,并标记 ModelAndView 已处理,这样就不会执行第 3 步的视图渲染  

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值