查看原文
其他

一个请求在Spring MVC中的执行过程是怎么样的?

点击上方蓝色“后端面试那些事儿”,选择“设为星标”

学最好的别人,做最好的我们

来源 | jianshu.com/p/e18fd44964eb



  • Spring MVC 过程
    • 1. 设置属性
    • 2. 根据 Request 请求的 URL 得到对应的 handler 执行链,其实就是拦截器和 Controller 代理对象。
    • 3. 得到 handler 的适配器
    • 4. 循环执行 handler 的 pre 拦截器
    • 5. 执行真正的 handler,并返回  ModelAndView(Handler 是个代理对象,可能会执行 AOP )
    • 6. 循环执行 handler 的 post 拦截器
    • 7. 根据 ModelAndView 信息得到 View 实例
    • 8. 渲染 View 返回

SpringMVC ,这个应该是国内面试必问题,网上有很多答案,其实背背就可以。但今天笔者带大家一起深入浅出源码,看看他的原理。以期让印象更加深刻,面试的时候游刃有余。

Spring MVC 过程

先来张图:

代码位置:com.interface21.web.servlet.DispatcherServlet#doService

(没错,就是 Spring 1.0 的代码,大道至简,现在的 Spring 经过 15 年的发展,已经太过臃肿,从学习角度来说,interface 21 是最好的代码,不接受反驳)

代码如下:

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 的适配器

// This will throw an exception if no adapter is found
// 3. 返回 handler 的适配器
HandlerAdapter ha = getHandlerAdapter(mappedHandler.getHandler());

关于这个适配器,作用到底是啥呢?HandlerAdapter 注释写到:This interface is not intended for application developers. It is available to handlers who want to develop their own web workflow. 译:此接口不适用于应用程序开发人员。它适用于想要开发自己的Web工作流程的处理程序。

也就说说,如果你想要在处理 handler 之前做一些操作的话,可能需要这个,即适配一下这个 handler。例如 Spring 的测试程序做的那样:

public ModelAndView handle(HttpServletRequest request, HttpServletResponse response, Object delegate)
            throws IOException, ServletException 
{
                      // 你可能需要 doSomething.......
            ((MyHandler) delegate).doSomething(request);
            return null;
        }

4. 循环执行 handler 的 pre 拦截器

// 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,并返回  ModelAndView(Handler 是个代理对象,可能会执行 AOP )

// 5. 执行真正的 handler,并返回  ModelAndView(Handler 是个代理对象,可能会执行 AOP )
ModelAndView mv = ha.handle(request, response, mappedHandler.getHandler());

6. 循环执行 handler 的 post 拦截器

// 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 返回

// 8. 渲染 View 返回
view.render(mv.getModel(), request, response);




往期推荐

Java反射是什么?

2020-06-25

消息队列常见的几种使用场景介绍!

2020-06-24

现在要做个点赞功能,你说说用 MySQL 还是 Redis 好 ?

2020-06-23

MySQL的explain 命令用过么?用来做啥?

2020-06-22

知道Java中的回调机制吗?

2020-06-21



一起进大厂,每日学干货

关注我,不迷路

    您可能也对以下帖子感兴趣

    文章有问题?点此查看未经处理的缓存