SpringMVC 核心源码分析

Spring的源码非常复杂,想要一篇概览几乎不可能,所以只能分而治之,本文摘写自近期抽空翻阅SpringMVC框架的部分源码,借此由浅入深地探索SpringMVC的核心体系结构。

在前后端分离大行其道的今日,由后端完成渲染的JSP/Freemarker时代可以说已经逐渐被NodeJS搭建的大前端体系取代,所以本文也不去翻阅SpringMVC视图渲染部分,仅认为后端作为一个数据服务接口来看到,简而言之,仅分析SpringMVC与前端的JSON交互流程。

基于过往的研发经验相信绝大多数都了解到,SpringMVC的核心组件为DispatcherServlet组件,而实现的规约则是J2EE关于Servlet的规范。如果对于Servlet规范不甚清楚,请抽空自行补充。本篇仅讨论HTTP请求的完整调用链路,不涉及其他协议规范说明。

DispatcherServlet

作为一个全局Servlet,欲深究其意,需先了解DispatcherServlet容器初始化时做了什么处理?而DispatcherServlet对象初始化包括如下三部分:

  • DispatcherServlet对象静态初始化部分?
  • 构造函数?
  • Servlet规范中容器初始化流程?

在整个容器初始化的过程,只有明白其中会涉及到得我们需要去核心关注的部分,才能很好的把握住SpringMVC的核心流程,从中透析核心体系。

静态初始化部分

static {
        // Load default strategy implementations from properties file.
        // This is currently strictly internal and not meant to be customized
        // by application developers.
        try {
            ClassPathResource resource = new ClassPathResource(DEFAULT_STRATEGIES_PATH, DispatcherServlet.class);
            defaultStrategies = PropertiesLoaderUtils.loadProperties(resource);
        }
        catch (IOException ex) {
            throw new IllegalStateException("Could not load '" + DEFAULT_STRATEGIES_PATH + "': " + ex.getMessage());
        }
    }

静态初始化实际是读取了DispatcherServlet.properties文件的内容,其中都有些什么呢?摘4.3.18.RELESE版本的文件,内容如下:

// 省略...
org.springframework.web.servlet.HandlerMapping=org.springframework.web.servlet.handler.BeanNameUrlHandlerMapping,\
    org.springframework.web.servlet.mvc.annotation.DefaultAnnotationHandlerMapping

org.springframework.web.servlet.HandlerAdapter=org.springframework.web.servlet.mvc.HttpRequestHandlerAdapter,\
    org.springframework.web.servlet.mvc.SimpleControllerHandlerAdapter,\
    org.springframework.web.servlet.mvc.annotation.AnnotationMethodHandlerAdapter
// 省略....

找一个类进去翻阅,可以猜测它可能是要初始化这些Bean,再查阅DispatcherServlet#getDefaultStrategies() 即可确认,当然,这部分并非是静态初始化就完成,而是延迟至Serlvet容器初始化时分批进行初始化。

构造函数

翻阅代码可以发现内部有两个构造函数,那么究竟在容器初始化的时候默认调用的是哪个?

easy,打个断点DEBUG一下就可以,基于我的环境上我发现是默认无参构造函数。(基于springboot 1.5.6版本)

那么是谁来调用改无参构造函数的呢?

easy,在springboot环境运行时可以看到有一个显眼的注解SpringBootApplication,而该注解的组成部分还包括EnableAutoConfiguration,这个注解又是要解决什么的呢?

它将默认加载spring-boot-autoconfig下的spring.factories文件内部的类,比如:

org.springframework.boot.autoconfigure.web.DispatcherServletAutoConfiguration,\
org.springframework.boot.autoconfigure.web.WebMvcAutoConfiguration,\

///.....

那么查阅这些类,它们做了什么?比如DispatcherServletAutoConfiguration,查阅时发现它加了注解Configuration

查阅该注解你又将发现它会处理声明@Bean的方法,并将该Bean注入到Spring容器中,由容器托管其生命周期。

Servlet容器初始化

查阅javax.servlet.Servlet接口的定义,可以清晰的看到规范中定义四个方法,其中需要重点关注如 init, service

那么DispatcherServlet容器在初始化时做了什么?

HttpServlet#init() -> FrameworkServlet#initServletBean -> DispatchserServlet#initStrategies()

注:如果想要知道Serlevt如何工作,可以翻阅Tomcat源码,以此看到工业级产品如何实现Servlet规范

翻阅DispatchserServlet#initStrategies()可以看到好几个方法,方法名称皆以init打头,那么它们就是在初始化什么?为什么在自己的程序中使用@Controller, @RequestMapping, @ResponseBody就可以跟前端完成常见的JSON交互了呢?

为了完成非常复杂的功能交互,SpringMVC定义了非常多的顶级接口,而我们核心要关注的无非是HandlerMapping, HandlerAdpater

HandlerMapping

查阅DispatcherServlet#initHandlerMappings()方法时发现一个问题,那就是它并非依据前文中提到的DispatcherServlet.properties中加载出来的对象,而是从ApplicationContext中取

Map matchingBeans =
                    BeanFactoryUtils.beansOfTypeIncludingAncestors(context, HandlerMapping.class, true, false);

那么取出了什么Bean呢?

SpringMVC 核心源码分析_第1张图片
image.png

这些Bean从而何来?前文说过,本文基于springboot 1.5.6版本,程序启动加载spring-boot-autoconfig下的spring.factories文件,其中比如org.springframework.boot.autoconfigure.web.WebMvcAutoConfiguration

由于我们实际编码过程经常使用@RequestMapping来完成URL路径标记,那么先从与之有些类似的RequestMappingHandlerMapping中翻阅代码,看看初始化时核心放了什么?

public RequestMappingHandlerMapping requestMappingHandlerMapping() {
        RequestMappingHandlerMapping mapping = createRequestMappingHandlerMapping();
        mapping.setOrder(0);
        mapping.setInterceptors(getInterceptors());
        mapping.setContentNegotiationManager(mvcContentNegotiationManager());
        mapping.setCorsConfigurations(getCorsConfigurations());

        PathMatchConfigurer configurer = getPathMatchConfigurer();
        if (configurer.isUseSuffixPatternMatch() != null) {
      // 比如这个,很多开发者应该都记得以前URL一般都有后缀,比如.action, .do, .ac等等
            mapping.setUseSuffixPatternMatch(configurer.isUseSuffixPatternMatch());
        }
        if (configurer.isUseRegisteredSuffixPatternMatch() != null) {
            mapping.setUseRegisteredSuffixPatternMatch(configurer.isUseRegisteredSuffixPatternMatch());
        }
        if (configurer.isUseTrailingSlashMatch() != null) {
            mapping.setUseTrailingSlashMatch(configurer.isUseTrailingSlashMatch());
        }
        UrlPathHelper pathHelper = configurer.getUrlPathHelper();
        if (pathHelper != null) {
            mapping.setUrlPathHelper(pathHelper);
        }
        PathMatcher pathMatcher = configurer.getPathMatcher();
        if (pathMatcher != null) {
            mapping.setPathMatcher(pathMatcher);
        }

        return mapping;
    }

单单查阅内部方法,类的命名大致可以猜测出一些信息来,不过最终的使用还是要通过其他核心流程来分析。

HandlerAdpater

同理的方式可以看到核心的关注类

image.png

Servlet容器工作流程

DispatcherServlet#doDispatch(),可以认为是Servlet规范中的service()语义。

截取核心代码如下,并且将会按照入口核心源码来分析

HandlerExecutionChain mappedHandler = getHandler(processedRequest);
HandlerAdapter ha = getHandlerAdapter(mappedHandler.getHandler());
ModelAndView mv = ha.handle(processedRequest, response, mappedHandler.getHandler());
processDispatchResult(processedRequest, response, mappedHandler, mv, dispatchException);

getHandler(processedRequest)

/**
     * Return the HandlerExecutionChain for this request.
     * 

Tries all handler mappings in order. * @param request current HTTP request * @return the HandlerExecutionChain, or {@code null} if no handler could be found */ protected HandlerExecutionChain getHandler(HttpServletRequest request) throws Exception { for (HandlerMapping hm : this.handlerMappings) { if (logger.isTraceEnabled()) { logger.trace( "Testing handler map [" + hm + "] in DispatcherServlet with name '" + getServletName() + "'"); } HandlerExecutionChain handler = hm.getHandler(request); if (handler != null) { return handler; } } return null; }

方法语义其实很简单,从注入的一堆HandlerMapping中查找合适的一个并返回。那么什么是合适的?查阅HandlerMapping中对于getHandler()方法的注解:

Return a handler and any interceptors for this request. The choice may be made
on request URL, session state, or any factor the implementing class chooses.

HandlerExecutionChain包含着真正的HandlerMapping, 以及一系列的拦截器。通过DEBUG可以发现最终找到的对象为RequestMappingHandlerMapping,怎么匹配的呢?通过结果来反向推导原因。Spring框架中大量使用了模板方法,所以看类实现过程要关注子类与父类之间的关系。

RequestMappingHandlerMapping
类体系

先查阅整个类结构体系,相当复杂,不过不要紧,看关键点。

HandlerMapping <-- AbstractHandlerMapping <-- AbstractHandlerMethodMapping <-- RequestMappingInfoHandlerMapping <--RequestMappingHandlerMapping

初始化流程

RequestMappingHandlerMapping作为一个由Spring托管的Bean,在它初始化的生命周期中,有非常关键的一步,即afterPropertiesSet方法,跟踪上去,最终你会看到如下几个核心的方法:

//初始化HandlerMethods
AbstractHandlerMethodMapping#initHandlerMethods
// 判断这个类是否有@Controller 或者@RequestMapping
RequestMappingHandlerMapping#isHandler()
// 查找hanlder内部方法
AbstractHandlerMethodMapping#detectHandlerMethods()
// 找到handler内部带有标记的@RequestMapping的方法,包装成RequestMappingInfo
RequestMappingHandlerMapping#getMappingForMethod()
// 找到一个可以被调用的目标方法Method
AopUtils.selectInvocableMethod(entry.getKey(), userType)
// 将相信的handler,method等信息注册到MappingRegistry中,注册前会将信息包装成HandlerMethod
AbstractHandlerMethodMapping#registerHandlerMethod(handler, method, RequestMappingInfo)
// 注册,MappingRegistry有非常核心的Map,比如urlLookup
MappingRegistry#register(RequestMappingInfo, handler, Method)
最终的HandlerMethod包括了什么信息?

this.bean = bean;
this.beanType = ClassUtils.getUserClass(bean);
this.method = method;
this.bridgedMethod = BridgeMethodResolver.findBridgedMethod(method);
this.parameters = initMethodParameters();

注:这些信息最终可以通过反射来完成方法的调用。

在完成初始化之后,当进来时,如何完成URLController中的Method的对应呢?

映射

AbstractHandlerMethodMapping#getHandlerInternal()
AbstractHandlerMethodMapping#lookupHandlerMethod()
MappingRegistry#getMappingsByUrl()
当完成初始化之后,通过请求的URL找到对应的HandlerMethod其实非常简单.

getHandlerAdapter()

通过RequestMappingHandlerMapping来找Adapter,先来看HandlerAdapter接口的定义:MVC framework SPI, allowing parameterization of the core MVC workflow.

它的核心在于,找到RequestMappingHandlerMapping来完成从请求到执行到响应的完整工作流程。

通过DEBUG可以非常快速定外找,需要关注的核心实现为RequestMappingHandlerAdapter.通过其父类实现的support()其实也可以非常快速确定,因为它支持的Handler类型为HandlerMethod.

handler()

RequestMappingHandlerAdapter

可以非常明显看到这个类内部包含着一系列的HttpMessageConverter,这些对象可以解决什么问题呢?

方法调用

RequestMappingHandlerAdapter#invokeHandlerMethod()
ServletInvocableHandlerMethod#invokeAndHandle()
InvocableHandlerMethod#invokeForRequest()
InvocableHandlerMethod#doInvoke()
// 最终方法执行
Method#invoke()

也就是说,HTTP请求进入SpringMVC应用之后,匹配到HandlerMethod,最终通过HandlerMethod中包含的Method对象,而Method通过Java反射即可完成方法调用。

完成调用的过程分两部分:

  • 从HTTP中解析参数
  • 将返回值进行处理(比如声明@ResponseBody
参数解析

参数的解析以及映射是一件相当复杂的工作,比如

@ResponseBody
@RequestMapping("data")
fun userData(@RequestParam TransferObject json) : UserData {
    return UserData(...);
}

在一个完整的HTTP请求中,核心参数可能在URL上,也可以在Body上,当要完成参数解析时,需要从中取出参数,并按照方法所需入参进行封装,最后入参完成方法调用,欲深究请关注如下方法

InvocableHandlerMethod#getMethodArgumentValues()

返回值处理

返回值可能会有各式各样的结果,一般情况下可能是一个JSON对象,通过DEBUG可以看到核心类RequestResponseBodyMethodProcessor

public boolean supportsReturnType(MethodParameter returnType) {
        return (AnnotatedElementUtils.hasAnnotation(returnType.getContainingClass(), ResponseBody.class) ||
                returnType.hasMethodAnnotation(ResponseBody.class));
    }

从HTTP协议来看,在入参时就已经明确了响应体的类型,核心参数为:accept,常见比如application/json, text/html, 等等。SpringMVC如何处理这个问题呢?请聚焦于MediaType

AbstractMessageConverterMethodProcessor#writeWithMessageConverters()

// 从http请求中取出来后进行解析
List requestedMediaTypes = getAcceptableMediaTypes(request);
// 根据返回值判断何时的MediaType
List producibleMediaTypes = getProducibleMediaTypes(request, valueType, declaredType);

比如上请求,根据返回值判断即为JSON,故而结果如下图:

image.png

MediaType将决定以何中HttpMessageConvert来完成对象的转换处理,以及流输出。

messageConverter.canWrite(declaredType, valueType, selectedMediaType)
messageConverter.write(outputValue, declaredType, selectedMediaType, outputMessage)

而判断的核心在于实现类的supportedMediaTypes参数值,比如:

public MappingJackson2HttpMessageConverter(ObjectMapper objectMapper) {
        super(objectMapper, MediaType.APPLICATION_JSON, new MediaType("application", "*+json"));
}
钩子函数

在将返回值写入到输出流中前,提供了钩子函数RequestResponseBodyAdviceChain

processDispatchResult()

完成请求的后置处理。

比如mappedHandler.triggerAfterCompletion(request, response, null)

其实是将所有的后置拦截器完成调用。

思考

看完了完整的流程后,如果需要对SpringMVC做一些优化,需要如何?

优化的层面往往是往下往上两个层面。

  • 往下从使用层面去考虑,选用更适合SpringMVC框架的用法,比如URL匹配规则简单化
  • 往上从优化层面去考虑,比如减少不必要的Bean创建,简化从HandlerMap, HandlerAdapter等的寻找链路,加快容器初始化,提高调用链路效率,等等

你可能感兴趣的:(SpringMVC 核心源码分析)