SpringMVC工作流程
参考文章:
https://www.toutiao.com/i6618344913593451016/
https://www.toutiao.com/i6613233781933343240/
简单来说:
全程由同一个线程执行。基本所有主要逻辑都在doDispatch()方法中完成。
客户端发送请求-> 前端控制器 DispatcherServlet 接受客户端请求 -> 找到处理器映射 HandlerMapping 解析请求对应的 Handler-> HandlerAdapter 会根据 Handler 来调用真正的处理器开处理请求,并处理相应的业务逻辑 -> 处理器返回一个模型视图 ModelAndView -> 视图解析器进行解析 -> 返回一个视图对象->前端控制器 DispatcherServlet 渲染数据(Moder)->将得到视图对象返回给用户。
进入的顺序是Filter-->Interceptor-->ControllerAdvice-->Aspect-->Controller(后进先出)
如下图所示:
代码如下:
1.设置属性
// 1. 设置属性
// Make web application context available
request.setAttribute(WEB_APPLICATION_CONTEXT_ATTRIBUTE, getWebApplicationContext());
// Make locale resolver available
request.setAttribute(LOCALE_RESOLVER_ATTRIBUTE, this.localeResolver);
// Make theme resolver available
request.setAttribute(THEME_RESOLVER_ATTRIBUTE, this.themeResolver);
2.根据 Request 请求的 URL 得到对应的 handler 执行链,其实就是拦截器和 Controller 代理对象。
// 2. 找 handler 返回执行链
HandlerExecutionChain mappedHandler = getHandler(request);
3.得到 handler 的适配器
此接口不适用于应用程序开发人员。它适用于想要开发自己的Web工作流程的处理程序。也就说,如果你想要在处理 handler 之前做一些操作的话,可能需要这个,即适配一下这个 handler。例如 Spring 的测试程序做的那样。
// This will throw an exception if no adapter is found
// 3. 返回 handler 的适配器
HandlerAdapter ha = getHandlerAdapter(mappedHandler.getHandler());
4.顺序执行 handler 的 pre 拦截器(按xml中拦截器配置顺序执行,参照6,后进先出 与AOP相同)
// 4. 循环执行 handler 的 pre 拦截器
for (int i = 0; i < mappedHandler.getInterceptors().length; i++) {
HandlerInterceptor interceptor = mappedHandler.getInterceptors()[i];
// pre 拦截器
if (!interceptor.preHandle(request, response, mappedHandler.getHandler())) {
return;
}
}
5.执行真正的 handler(controller),并返回 ModelAndView(Handler 是个代理对象,可能会执行 AOP )
消息转换:json/xml转Object
数据转换:如string转integer等
数据格式化:如string格式化
数据验证:如JSR303
// 5. 执行真正的 handler,并返回 ModelAndView(Handler 是个代理对象,可能会执行 AOP )
ModelAndView mv = ha.handle(request, response, mappedHandler.getHandler());
6.倒序执行 handler 的 post 拦截器(按xml中拦截器配置顺序倒序执行)
// 6. 循环执行 handler 的 post 拦截器
for (int i = mappedHandler.getInterceptors().length - 1; i >=0 ; i--) {
HandlerInterceptor interceptor = mappedHandler.getInterceptors()[i];
// post 拦截器
interceptor.postHandle(request, response, mappedHandler.getHandler());
}
7.根据 ModelAndView 信息得到 View 实例
View view = null;
if (mv.isReference()) {
// We need to resolve this view name
// 7. 根据 ModelAndView 信息得到 View 实例
view = this.viewResolver.resolveViewName(mv.getViewName(), locale);
}
8.渲染 View 返回