阅读源码有利于陶冶情操,此文承接前文SpringMVC源码情操陶冶-DispatcherServlet父类简析

注意:springmvc初始化其他内容,其对应的配置文件已被加载至beanFactory中

DispatcherServlet#静态方法

DispatcherServlet静态块读取DispatcherServlet.properties,其中包含默认的策略实现

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 'DispatcherServlet.properties': " + ex.getMessage());
}
}

简单看下DispatcherServlet.properties文件,具体内容如下

# Default implementation classes for DispatcherServlet's strategy interfaces.
# Used as fallback when no matching beans are found in the DispatcherServlet context.
# Not meant to be customized by application developers. org.springframework.web.servlet.LocaleResolver=org.springframework.web.servlet.i18n.AcceptHeaderLocaleResolver org.springframework.web.servlet.ThemeResolver=org.springframework.web.servlet.theme.FixedThemeResolver #url映射
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 #异常解析
org.springframework.web.servlet.HandlerExceptionResolver=org.springframework.web.servlet.mvc.annotation.AnnotationMethodHandlerExceptionResolver,\
org.springframework.web.servlet.mvc.annotation.ResponseStatusExceptionResolver,\
org.springframework.web.servlet.mvc.support.DefaultHandlerExceptionResolver org.springframework.web.servlet.RequestToViewNameTranslator=org.springframework.web.servlet.view.DefaultRequestToViewNameTranslator #视图解析
org.springframework.web.servlet.ViewResolver=org.springframework.web.servlet.view.InternalResourceViewResolver org.springframework.web.servlet.FlashMapManager=org.springframework.web.servlet.support.SessionFlashMapManager

DispatcherServlet#onRefresh

复写父类FrameworkServlet的onRefresh()方法,springmvc以此来完善相应的初始化操作,代码如下

	//复写
@Override
protected void onRefresh(ApplicationContext context) {
initStrategies(context);
}

直接看DispatcherServlet#initStrategies()初始化springmvc的策略对象,代码如下

protected void initStrategies(ApplicationContext context) {
initMultipartResolver(context);
initLocaleResolver(context);
initThemeResolver(context);
initHandlerMappings(context);
initHandlerAdapters(context);
initHandlerExceptionResolvers(context);
initRequestToViewNameTranslator(context);
initViewResolvers(context);
initFlashMapManager(context);
}

以上述程序的执行顺序,逐个分析

DispatcherServlet#initMultipartResolver(ApplicationContext context)

根据官方注释,此会去加载beanFactory的multipartResolverbean对象,适用于处理文件上传的操作,简单代码如下

private void initMultipartResolver(ApplicationContext context) {
try {
this.multipartResolver = context.getBean(MULTIPART_RESOLVER_BEAN_NAME, MultipartResolver.class);
if (logger.isDebugEnabled()) {
logger.debug("Using MultipartResolver [" + this.multipartResolver + "]");
}
}
catch (NoSuchBeanDefinitionException ex) {
// Default is no multipart resolver.默认是没有对应的文件上传处理类的
this.multipartResolver = null;
if (logger.isDebugEnabled()) {
logger.debug("Unable to locate MultipartResolver with name '" + MULTIPART_RESOLVER_BEAN_NAME +
"': no multipart request handling provided");
}
}
}

DispatcherServlet#initLocaleResolver(ApplicationContext context)

本地校对解析,官方解释为获取id="localeResolver"的bean对象,如果没有则使用默认的org.springframework.web.servlet.i18n.AcceptHeaderLocaleResolver类创建bean对象

private void initLocaleResolver(ApplicationContext context) {
try {
this.localeResolver = context.getBean(LOCALE_RESOLVER_BEAN_NAME, LocaleResolver.class);
if (logger.isDebugEnabled()) {
logger.debug("Using LocaleResolver [" + this.localeResolver + "]");
}
}
catch (NoSuchBeanDefinitionException ex) {
// We need to use the default.
this.localeResolver = getDefaultStrategy(context, LocaleResolver.class);
if (logger.isDebugEnabled()) {
logger.debug("Unable to locate LocaleResolver with name '" + LOCALE_RESOLVER_BEAN_NAME +
"': using default [" + this.localeResolver + "]");
}
}
}

Note:这里我们简单看下DispatcherServlet#getDefaultStrategy()的内部方法createDefaultStrategy()创建spring bean对象

protected Object createDefaultStrategy(ApplicationContext context, Class<?> clazz) {

//可通过ApplicationContext.getAutowireCapableBeanFactory()方法来额外的创建外部class为bean对象

return context.getAutowireCapableBeanFactory().createBean(clazz);

}

DispatcherServlet#initThemeResolver(ApplicationContext context)

DispatcherServlet#initLocaleResolver(ApplicationContext context),找寻id="themeResolver"的bean对象,没有则默认使用org.springframework.web.servlet.theme.FixedThemeResolver作为bean对象

DispatcherServlet#initHandlerMappings(ApplicationContext context)

初始化对资源访问的映射处理类,代码如下

private void initHandlerMappings(ApplicationContext context) {
this.handlerMappings = null;
/***默认为true,表示从beanFactory和其父类获取所有的HandlerMapping
** beans
**<init-param>
* <param-name>detectAllHandlerMappings</param-name>
* <param-value>true/false</param-value>
*/
if (this.detectAllHandlerMappings) {
//从祖先类到子类都去找寻相应的资源
Map<String, HandlerMapping> matchingBeans =
BeanFactoryUtils.beansOfTypeIncludingAncestors(context, HandlerMapping.class, true, false);
if (!matchingBeans.isEmpty()) {
this.handlerMappings = new ArrayList<HandlerMapping>(matchingBeans.values());
// We keep HandlerMappings in sorted order.
//根据优先级排序
OrderComparator.sort(this.handlerMappings);
}
}
else {
try {
//否则获取本springmvc的上下文中的单个`id="handlerMapping"`的HandlerMapping bean
HandlerMapping hm = context.getBean(HANDLER_MAPPING_BEAN_NAME, HandlerMapping.class);
this.handlerMappings = Collections.singletonList(hm);
}
catch (NoSuchBeanDefinitionException ex) {
// Ignore, we'll add a default HandlerMapping later.
}
} //如果detectAllHandlerMappings=false且springmvc不包含相应的`handlerMapping`bean 则取默认的HandlerMappings
if (this.handlerMappings == null) {
this.handlerMappings = getDefaultStrategies(context, HandlerMapping.class);
if (logger.isDebugEnabled()) {
logger.debug("No HandlerMappings found in servlet '" + getServletName() + "': using default");
}
}
}

默认的HandlerMappings为org.springframework.web.servlet.handler.BeanNameUrlHandlerMapping

org.springframework.web.servlet.mvc.annotation.DefaultAnnotationHandlerMapping,从字面上可知前者是根据beanName直接映射请求,后者是注解方式的映射请求

DispatcherServlet#initHandlerAdapters(ApplicationContext context)

此处类似DispatcherServlet#initHandlerMapping()

  1. detectAllHandlerAdapters默认为true,也可通过web.xml修改值,表示找寻所有满足HandlerAdapter.class的beans;若为false,则寻找springmvc本身上下文环境(springmvc配置文件)的id="handlerAdapter"的单个bean对象。所以不管如何,springmvc若指定了handlerAdapterbean,其总是会被加载

  2. 默认的策略对象为SimpleControllerHandlerAdapter-简单的Controller请求处理

DispatcherServlet#initHandlerExceptionResolvers(ApplicationContext context)

此处类似DispatcherServlet#initHandlerMapping()

  1. detectAllHandlerExceptionResolvers默认为true,也可通过web.xml修改值,表示找寻所有满足HandlerExceptionResolver.class的beans;若为false,则寻找springmvc本身上下文环境(springmvc配置文件)的id="handlerExceptionResolver"的单个bean对象。所以不管如何,springmvc若指定了handlerExceptionResolverbean,其总是会被加载

  2. 默认的策略对象为SimpleControllerHandlerAdapter-简单的Controller请求处理

DispatcherServlet#initRequestToViewNameTranslator(ApplicationContext context)

此处同DispatcherServlet#initLocaleResolver()方法

  1. 首先找寻springmvc上下文中的id="viewNameTranslator"的bean,类型为RequestToViewNameTranslator.class
  2. 不然则采用默认的策略对象org.springframework.web.servlet.view.DefaultRequestToViewNameTranslator-对请求的URI直接映射到view对象

DispatcherServlet#initViewResolvers(ApplicationContext context)

此处类似DispatcherServlet#initViewResolvers()

  1. detectAllViewResolvers默认为true,也可通过web.xml修改值,表示找寻所有满足ViewResolver.class的beans;若为false,则寻找springmvc本身上下文环境(springmvc配置文件)的id="viewResolver"的单个bean对象。所以不管如何,springmvc若指定了viewResolverbean,其总是会被加载

  2. 默认的策略对象为InternalResourceViewResolver-默认解析为jsp视图

DispatcherServlet#initFlashMapManager(ApplicationContext context)

此处同DispatcherServlet#initLocaleResolver()方法

  1. 首先找寻springmvc上下文中的id="flashMapManager"的bean,类型为FlashMapManager.class
  2. 不然则采用默认的策略对象org.springframework.web.servlet.support.SessionFlashMapManager-保存并检索FlashMap属性

小结

本文只介绍springmvc额外的初始化操作,包括文件上传处理类、请求映射处理类/适配器、异常处理类、视图解析类等等

SpringMVC源码情操陶冶-DispatcherServlet类简析(一)的更多相关文章

  1. SpringMVC源码情操陶冶-DispatcherServlet父类简析

    阅读源码有助于陶冶情操,本文对springmvc作个简单的向导 springmvc-web.xml配置 <servlet> <servlet-name>dispatch< ...

  2. SpringMVC源码情操陶冶-HandlerAdapter适配器简析

    springmvc中对业务的具体处理是通过HandlerAdapter适配器操作的 HandlerAdapter接口方法 列表如下 /** * Given a handler instance, re ...

  3. SpringMVC源码情操陶冶-DispatcherServlet简析(二)

    承接前文SpringMVC源码情操陶冶-DispatcherServlet类简析(一),主要讲述初始化的操作,本文将简单介绍springmvc如何处理请求 DispatcherServlet#doDi ...

  4. SpringMVC源码情操陶冶-DispatcherServlet

    本文对springmvc核心类DispatcherServlet作下简单的向导,方便博主与读者查阅 DispatcherServlet-继承关系 分析DispatcherServlet的继承关系以及主 ...

  5. SpringMVC源码情操陶冶-RequestMappingHandlerAdapter适配器

    承接前文SpringMVC源码情操陶冶-HandlerAdapter适配器简析.RequestMappingHandlerAdapter适配器组件是专门处理RequestMappingHandlerM ...

  6. SpringMVC源码情操陶冶-FreeMarker之web配置

    前言:本文不讲解FreeMarkerView视图的相关配置,其配置基本由FreeMarkerViewResolver实现,具体可参考>>>SpringMVC源码情操陶冶-ViewRe ...

  7. SpringMVC源码情操陶冶-AnnotationDrivenBeanDefinitionParser注解解析器

    mvc:annotation-driven节点的解析器,是springmvc的核心解析器 官方注释 Open Declaration org.springframework.web.servlet.c ...

  8. SpringMVC源码情操陶冶-AbstractHandlerMethodMapping

    承接前文SpringMVC源码情操陶冶-AbstractHandlerMapping,本文将介绍如何注册HandlerMethod对象作为handler 类结构瞧一瞧 public abstract ...

  9. SpringMVC源码情操陶冶-AbstractUrlHandlerMapping

    承接前文SpringMVC源码情操陶冶-AbstractHandlerMapping,前文主要讲解了如何获取handler处理对象,本文将针对beanName注册为handler对象作下解析 Abst ...

随机推荐

  1. 【Java学习笔记之十九】super在Java继承中的用法小结

    1)有人写了个很好的初始化属性的构造函数,而你仅仅想要在其中添加另一些自己新建属性的初始化,这样在一个构造函数中调用另外一个构造函数,可以避免重复的代码量,减少工作量: 2)在一个构造函数中调用另外一 ...

  2. COGS 144. [USACO Dec07] 魅力手镯【01背包复习】

    144. [USACO Dec07] 魅力手镯 ★   输入文件:charm.in   输出文件:charm.out   简单对比 时间限制:1 s   内存限制:8 MB 译 by CmYkRgB1 ...

  3. 4层板的pcb创建

    四层板的制作可以在原先的二层板上增加层,(软件为AD17)方法如下: 一.单击Design->Layer Stack Manager菜单(快捷键为DK),将弹出以下的界面: 二,选择上图中的Ad ...

  4. RxJava整合Retrofit遇到的问题总结

    一:初上手(填坑) Observable将事件序列执行完毕后,会回调Observe的onNext()方法和onCompleted()方法,当出现异常/错误时会调用onError()方法. 由此,我们推 ...

  5. VUE之随笔小总结1

    VUE 它是一个构建用户界面的JavaScript框架vue指令: 是带有v-前缀的特殊属性,通过属性来操作元素 v-text:在元素当中插入文本 eg:属性值会覆盖自己插入的值 //插入一段文本&l ...

  6. Spark算子--map和flatMap

    map和flatMap--Transformation类算子 代码示例 result  

  7. Binlog的三个业务应用场景

    1.什么是binlog binlog是mysql的一种二进制日志文件,用来记录数据的变化.mysql使用binlog进行主从复制,如图: 客户端向master的mysql sever写入数据 当数据发 ...

  8. Node.js进阶:5分钟入门非对称加密方法

    前言 刚回答了SegmentFault上一个兄弟提的问题<非对称解密出错>.这个属于Node.js在安全上的应用,遇到同样问题的人应该不少,基于回答的问题,这里简单总结下. 非对称加密的理 ...

  9. 5分钟把任意网站变成桌面软件--windows版

    本文源自于segmentfault的一篇专栏文章:https://segmentfault.com/a/1190000012924855  只不过这篇是MAC版本的,所以我试了下windows版的: ...

  10. Java XML 序列化和反序列化

    Utils 类: import java.io.FileInputStream; import java.io.FileNotFoundException; import java.io.FileWr ...