SpringMVC 的自动配置

详解 SpringMVC 的自动配置(SpringBoot 环境)


前面的文章中也有提到过 Spring Boot 中的自动配置原理,在实际开发中如果对这些默认配置不熟悉的话很难做到“灵活”,什么都只能听“人家的”,所以,在此就借 SpringMVC 来开刀,揭开 Spring Boot 自动配置的原理。

在官网上提到 SpringMVC 的内容其实少的可伶,有兴趣也可以过去传送过去。

https://docs.spring.io/spring-boot/docs/2.1.3.RELEASE/reference/htmlsingle/#boot-features-spring-mvc-auto-configuration

废话不多说,下面正式来介绍自动配置。

1.自动配置SpringMVC的视图解析器

Inclusion of ContentNegotiatingViewResolver and BeanNameViewResolver beans.—— 摘自官网

这里主要说的是自动配置了 ViewResolver (视图解析器:解析视图,说白了就是根据方法的返回值得到视图对象,视图对象决定如何再去进行下一步的处理,是转发还是重定向。)

上面提到的 ContentNegotiatingViewResolver 的作用就是整合所有的视图解析器(后面进行验证),如果我们需要自定义视图解析器,则直接添加到容器中即可,它会给我们自动组合起来。

分析:

所有的 web 配置都是由 WebMvcAutoConfiguration 这个类来完成的,所以,先从这个类下手。该类中有一个方法 viewResolver,方法的返回值就是上面提到的 ContentNegotiatingViewResolver ,还要注意该方法上有一个 @Bean 注解:

 @Bean
 @ConditionalOnBean(ViewResolver.class)
 @ConditionalOnMissingBean(name = "viewResolver", value = ContentNegotiatingViewResolver.class)
 public ContentNegotiatingViewResolver viewResolver(BeanFactory beanFactory) {
   ContentNegotiatingViewResolver resolver = new ContentNegotiatingViewResolver();
   resolver.setContentNegotiationManager(
   beanFactory.getBean(ContentNegotiationManager.class));
   // ContentNegotiatingViewResolver uses all the other view resolvers to locate
   // a view so it should have a high precedence
   resolver.setOrder(Ordered.HIGHEST_PRECEDENCE);
   return resolver;
 }

所以说,不用细看这个方法的逻辑就能判断这个方法的作用就是给容器中注册一个 ContentNegotiatingViewResolver 。既然这玩意是一个视图解析器,那么就得解析视图,所以进入这个类,找到解析视图的方法 resolveViewName

 @Override
 @Nullable
 public View resolveViewName(String viewName, Locale locale) throws Exception {
   RequestAttributes attrs = RequestContextHolder.getRequestAttributes();
   Assert.state(attrs instanceof ServletRequestAttributes, "No current ServletRequestAttributes");
   List<MediaType> requestedMediaTypes = getMediaTypes(((ServletRequestAttributes) attrs).getRequest());
   if (requestedMediaTypes != null) {
     //获取获选视图
     List<View> candidateViews = getCandidateViews(viewName, locale, requestedMediaTypes);
     //选择适合的视图对象
     View bestView = getBestView(candidateViews, requestedMediaTypes, attrs);
   if (bestView != null) {
     //返回最适合的视图对象
     return bestView;
   }
 }
​
 String mediaTypeInfo = logger.isDebugEnabled() && requestedMediaTypes != null ?
 " given " + requestedMediaTypes.toString() : "";
​
 if (this.useNotAcceptableStatusCode) {
   if (logger.isDebugEnabled()) {
   logger.debug("Using 406 NOT_ACCEPTABLE" + mediaTypeInfo);
   }
   return NOT_ACCEPTABLE_VIEW;
 }
 else {
   logger.debug("View remains unresolved" + mediaTypeInfo);
   return null;
  }
  }

注意上面的几行注释,也许你会有好奇,怎么去选择适合的视图对象呢?不慌,点进去 getCandidateViews 方法:

private List<View> getCandidateViews(String viewName, Locale locale, List<MediaType> requestedMediaTypes)
 throws Exception {
​
   List<View> candidateViews = new ArrayList<>();
   if (this.viewResolvers != null) {
     Assert.state(this.contentNegotiationManager != null, "No ContentNegotiationManager set");
     for (ViewResolver viewResolver : this.viewResolvers) {
       View view = viewResolver.resolveViewName(viewName, locale);
       if (view != null) {
         candidateViews.add(view);
       }
     for (MediaType requestedMediaType : requestedMediaTypes) {
       List<String> extensions = this.contentNegotiationManager.resolveFileExtensions(requestedMediaType);
       for (String extension : extensions) {
       String viewNameWithExtension = viewName + '.' + extension;
       view = viewResolver.resolveViewName(viewNameWithExtension, locale);
         if (view != null) {
           candidateViews.add(view);
         }
       }
     }
   }
 }
   if (!CollectionUtils.isEmpty(this.defaultViews)) {
     candidateViews.addAll(this.defaultViews);
   }
   return candidateViews;
 }

这几个意思?看到一个 List 容器(装候选视图的),然后遍历,没错,正是逐一去试。把所有视图解析器拿过来,挨个去解析视图,看看哪个最合适。那么这些视图解析器又是从何而来的呢?找到该类中的一个方法:

protected void initServletContext(ServletContext servletContext) {
 Collection<ViewResolver> matchingBeans =
   //利用一个工具从容器中获取所有的视图解析器
   BeanFactoryUtils.beansOfTypeIncludingAncestors(obtainApplicationContext(),  ViewResolver.class).values();
   if (this.viewResolvers == null) {
     this.viewResolvers = new ArrayList<>(matchingBeans.size());
     for (ViewResolver viewResolver : matchingBeans) {
       if (this != viewResolver) {
         this.viewResolvers.add(viewResolver);
       }
     }
   }
   else {
     for (int i = 0; i < this.viewResolvers.size(); i++) {
       ViewResolver vr = this.viewResolvers.get(i);
       if (matchingBeans.contains(vr)) {
         continue;
       }
 String name = vr.getClass().getName() + i;
 obtainApplicationContext().getAutowireCapableBeanFactory().initializeBean(vr, name);
     }
​
   }
   AnnotationAwareOrderComparator.sort(this.viewResolvers);
   this.cnmFactoryBean.setServletContext(servletContext);
 }

利用一个 BeanFactoryUtils 工具从容器中获取所有的视图解析器。哎哟呵,既然你是从容器中拿的,那么就刺激了,如果我们需要自定义视图解析器,只需要把我们定义的视图解析器往容器一丢,完事! 那么事实到底是不是我们猜想的这样的呢?接下来进入验证时刻(有没有一点蒋昌建的味道):

 //1.定义一个视图解析器
 private static class MyViewResolver implements ViewResolver {
​
 @Override
 public View resolveViewName(String viewName, Locale locale) throws Exception {
 return null;
 }
 }

//2.丢进容器中
 @Bean
 public ViewResolver viewResolver(){
 return new MyViewResolver();
 }

3.在核心控制器 DispatcherServlet 中找到 doDispatch 方法,并在方法上打上断点,

然后 debug 随便发起一个请求,发会看到上面的结果:咱们自定义的视图解析器就给管理了。所以,需要自定义视图解析就直接将自定义的解析器丢进容器中。

2. 类型转换器以及格式化器

Automatic registration of Converter, GenericConverter, and Formatter beans —— 摘自官网

大致意思就是自动注册了转换器和日期格式化器。套路还是一样,先从最初的 WebMvcAutoConfiguration 开始。然后找到 addFormatter 方法,接着你会发现,这玩意又是在容器中取的,套路都是一样的,这里就不作详细的分析了。直接给出结论:

无论是转换器还是格式化器,如果需要自定制,则直接丢容器里面即可。

SpringMVC 给我们配置其实还不仅仅是上面提到的这两个(对于静态资源的配置前面的文章已经提到过了),还有好一大堆的呢,如果感兴趣,可以从这 org.springframework.boot.autoconfigure.web 刨出来看看。

3. 关于SpringMVC 默认配置的原则

Spring Boot 会先看容器中有没有用户自定义的配置,如果有,则使用用户配置的,如果没有,才使用默认配置的;如果有些组件可以有多个(比如上面提到的 ViewResolver ),将用户配置的和默认配置的结合起来。

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 214,776评论 6 496
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,527评论 3 389
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 160,361评论 0 350
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,430评论 1 288
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,511评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,544评论 1 293
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,561评论 3 414
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,315评论 0 270
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,763评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,070评论 2 330
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,235评论 1 343
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,911评论 5 338
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,554评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,173评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,424评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,106评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,103评论 2 352

推荐阅读更多精彩内容