前往小程序,Get更优阅读体验!
立即前往
首页
学习
活动
专区
工具
TVP
发布
社区首页 >专栏 >Spring MVC 的执行过程,面试必问了!

Spring MVC 的执行过程,面试必问了!

作者头像
Java技术栈
发布2020-11-23 15:37:04
4580
发布2020-11-23 15:37:04
举报
文章被收录于专栏:Java技术栈Java技术栈

关注阅读更多优质文章

Spring AOP、Spring MVC 这两个框架应该是国内面试必问的题目了,当然,网上有很多答案,其实背背就可以了。但如果你想看系列 Spring 面试题,可以关注公众号Java技术栈回复面试。

今天就带大家一起深入浅出源码,看看它的底层原理,可以让印象更加深刻,面试的时候也会游刃有余。

Spring AOP 原理

简单说说 AOP 的设计:

  1. 每个 Bean 都会被 JDK 或者 Cglib 代理,这取决于是否有接口。
  2. 每个 Bean 会有多个“方法拦截器”。注意:拦截器分为两层,外层由 Spring 内核控制流程,内层拦截器是用户设置,也就是 AOP。
  3. 当代理方法被调用时,先经过外层拦截器,外层拦截器根据方法的各种信息判断该方法应该执行哪些“内层拦截器”。内层拦截器的设计就是职责连的设计。

是不是很简单??

事实上,我之前已经写过一个简单的例子,看完之后更简单。

继续来分析。。。 第一:代理的创建; 第二:代理的调用。

注意:我们应该尽量少贴代码,应该尽量用文字叙述,因为面试的时候也是要用文字叙述,不可能让你把代码翻出来的。

所以,一定要保持一定的简洁,想知道细节,请看 interface 21 的源码,想知道的更细的,可以看 Spring Framework 框架的最新 master 分支代码。

代码的位置在:com.interface21.aop 包下面。

下面开始分析:

  1. 代理的创建(按步骤):
  • 首先需要创建代理工厂,代理工厂需要 3 个重要的信息,它们分别是拦截器数组,目标对象接口数组,目标对象。
  • 创建代理工厂时,默认会在拦截器数组尾部再增加一个默认拦截器,这是用于最终的调用目标方法。
  • 然后当调用 getProxy 方法的时候,会根据接口数量大余 0 条件返回一个代理对象(JDK or Cglib都可以)。
  • 注意:创建代理对象的同时会创建一个外层拦截器,这个拦截器就是 Spring 内核的拦截器,它用于控制整个 AOP 的流程。
  1. 代理的调用
  • 当对代理对象进行调用时,就会触发外层拦截器。
  • 外层拦截器根据代理配置信息,创建内层拦截器链。创建的过程中会根据表达式判断当前拦截是否匹配这个拦截器,而这个拦截器链设计模式就是职责链模式。
  • 当整个链条执行到最后时,就会触发创建代理时那个尾部的默认拦截器,从而调用目标方法。最后返回。

另外,这是一个题外话:Spring 的事务也就是个拦截器。关注公众号Java技术栈回复spring可以获取我整理的 Spring 系列教程。

来张不是很标准的 UML 图:

关于调用过程,来张流程图:

大概就是这样子,具体更多的细节,请看源码,如果还不是很明白的话,请咨询本人,本人不确定这个图是否画的很浅显易懂 —— 最起码萌新看得懂才能称之为浅显易懂。

Spring MVC 过程

先来张图:

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

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

代码如下:

1. 设置属性
代码语言:javascript
复制
// 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 代理对象。
代码语言:javascript
复制
// 2. 找 handler 返回执行链
HandlerExecutionChain mappedHandler = getHandler(request);
3. 得到 handler 的适配器
代码语言:javascript
复制
// 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 的测试程序做的那样:

代码语言:javascript
复制
public ModelAndView handle(HttpServletRequest request, HttpServletResponse response, Object delegate)
    throws IOException, ServletException {
              // 你可能需要 doSomething.......
    ((MyHandler) delegate).doSomething(request);
    return null;
}
4. 循环执行 handler 的 pre 拦截器
代码语言:javascript
复制
// 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 )
代码语言:javascript
复制
// 5. 执行真正的 handler,并返回  ModelAndView(Handler 是个代理对象,可能会执行 AOP )
ModelAndView mv = ha.handle(request, response, mappedHandler.getHandler());
6. 循环执行 handler 的 post 拦截器
代码语言:javascript
复制
// 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 实例
代码语言:javascript
复制
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 返回
代码语言:javascript
复制
// 8. 渲染 View 返回
view.render(mv.getModel(), request, response);

作者:莫那一鲁道 链接:https://www.jianshu.com/p/e18fd44964eb

本文参与 腾讯云自媒体分享计划,分享自微信公众号。
原始发表:2020-11-20,如有侵权请联系 cloudcommunity@tencent.com 删除

本文分享自 Java技术栈 微信公众号,前往查看

如有侵权,请联系 cloudcommunity@tencent.com 删除。

本文参与 腾讯云自媒体分享计划  ,欢迎热爱写作的你一起参与!

评论
登录后参与评论
0 条评论
热度
最新
推荐阅读
目录
  • Spring AOP 原理
  • 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 返回
                  领券
                  问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档