当浏览器发送一个请求 http://localhost:8080/hello
后,请求到达服务器,其处理流程是:
-
服务器提供了 DispatcherServlet,它使用的是标准 Servlet 技术
-
路径:默认映射路径为
/
,即会匹配到所有请求 URL,可作为请求的统一入口,也被称之为前控制器-
jsp 不会匹配到 DispatcherServlet
-
其它有路径的 Servlet 匹配优先级也高于 DispatcherServlet
-
-
创建:在 Boot 中,由 DispatcherServletAutoConfiguration 这个自动配置类提供 DispatcherServlet 的 bean
-
初始化:DispatcherServlet 初始化时会优先到容器里寻找各种组件,作为它的成员变量
-
HandlerMapping,初始化时记录映射关系
-
HandlerAdapter,初始化时准备参数解析器、返回值处理器、消息转换器
-
HandlerExceptionResolver,初始化时准备参数解析器、返回值处理器、消息转换器
-
ViewResolver
-
-
-
DispatcherServlet 会利用 RequestMappingHandlerMapping 查找控制器方法
-
例如根据 /hello 路径找到 @RequestMapping("/hello") 对应的控制器方法
-
控制器方法会被封装为 HandlerMethod 对象,并结合匹配到的拦截器一起返回给 DispatcherServlet
-
HandlerMethod 和拦截器合在一起称为 HandlerExecutionChain(调用链)对象
-
-
DispatcherServlet 接下来会:
-
调用拦截器的 preHandle 方法
-
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 步走视图解析及渲染流程
-
-
-
调用拦截器的 postHandle 方法
-
处理异常或视图渲染
-
如果 1~3 出现异常,走 ExceptionHandlerExceptionResolver 处理异常流程
-
@ControllerAdvice 全局增强点5️⃣:@ExceptionHandler 异常处理
-
-
正常,走视图解析及渲染流程
-
-
调用拦截器的 afterCompletion 方法
-
初始化阶段
-
在 Web 容器第一次用到 DispatcherServlet 的时候,会创建其对象(在spring和springMVC整合的时候是由tomcat初始化并创建的,到了springboot的时候就是由spring创建的)并执行 init 方法
-
init 方法内会创建 Spring Web 容器,并调用容器 refresh 方法
-
refresh 过程中会创建并初始化 SpringMVC 中的重要组件, 例如 MultipartResolver(文件上传、表单处理),HandlerMapping(负责请求映射),HandlerAdapter(调用控制器的方法),HandlerExceptionResolver(处理异常)、ViewResolver (把字符串解析成视图对象)等
-
容器初始化后,会将上一步初始化好的重要组件,赋值给 DispatcherServlet 的成员变量,留待后用
匹配阶段
-
用户发送的请求统一到达前端控制器 DispatcherServlet(虽然是所有请求的入口,但不处理具体请求)
-
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 步的视图渲染