在springboot中添加mvc功能的正确姿势讲解

springboot 添加mvc功能

先放出来几个类(包含注解或接口)来观摩一下

  • WebMvcConfigurer
  • @EnableWebMvc
  • WebMvcConfigurerAdapter(已过时,不再详述,可以理解为继承该类有和实现WebMvcConfigurer一样的效果)
  • WebMvcConfigurationSupport
  • WebApplicationInitializer

这里只聊springboot或者无web.xml环境的情况,无论如何得看一下这个祖宗,以下代码来源于spring官网

public class MyWebApplicationInitializer implements WebApplicationInitializer {
    @Override
    public void onStartup(ServletContext servletCxt) {
        // Load Spring web application configuration
        AnnotationConfigWebApplicationContext ac = new AnnotationConfigWebApplicationContext();
        ac.register(AppConfig.class);
        ac.refresh();

        // Create and register the DispatcherServlet
        DispatcherServlet servlet = new DispatcherServlet(ac);
        ServletRegistration.Dynamic registration = servletCxt.addServlet("app", servlet);
        registration.setLoadOnStartup(1);
        registration.addMapping("/app/*");
    }
}

这个是一切无配置文件spring和springmvc整合的基础,用来替代原始的web.xml中的ContextLoadListener和DispatcherServlet,两者效果等同;

现在我们先基于上述代码的情况来说,请注意以下结论的前提是基于上述示例代码,很重要,其实就像是很久之前我们从零开始搭建整合方案一样,现在只是配置了整合的类,还没有功能,如果我们想要再配置一个json的消息转换器

那么我们就会有如下几种方案

  • 继承WebMvcConfigurationSupport
  • 实现WebMvcConfigurer
  • 继承WebMvcConfigurerAdapter(已过时不再详述)

继承WebMvcConfigurationSupport和实现WebMvcConfigurer的区别如下

  • WebMvcConfigurationSupport直接继承并使用@Configuration标识即可,而实现WebMvcConfigurer则需要标识为注解@Configuration以外还需要使用注解@EnableWebMvc标识才可,所以一个项目中可以有多个地方去实现这个接口,只要标识为配置类。然后在一处地方开启@EnableWebMvc就可。这里提前说一下如果是springboot环境这里还大有说头,这里还有个大坑,留在后面说
  • WebMvcConfigurationSupport更偏向底层,可以定制化的功能更多,而WebMvcConfigurer是一个接口,是针对WebMvcConfigurationSupport功能将一些常用的功能选择性的暴露出来,实际上WebMvcConfigurer是依赖于WebMvcConfigurationSupport来实现功能添加的

为什么说WebMvcConfigurer是依赖于WebMvcConfigurationSupport来实现功能添加的?我们来看一下配合该接口的注解@EnableWebMvc源码

@Retention(RetentionPolicy.RUNTIME)
@Target(ElementType.TYPE)
@Documented
@Import(DelegatingWebMvcConfiguration.class)
public @interface EnableWebMvc {
}

来看一下这个注解导入的配置类DelegatingWebMvcConfiguration为何物?以下摘取该类部分源码

@Configuration
public class DelegatingWebMvcConfiguration extends WebMvcConfigurationSupport {
 private final WebMvcConfigurerComposite configurers = new WebMvcConfigurerComposite();
 @Autowired(required = false)
 public void setConfigurers(List<WebMvcConfigurer> configurers) {
  if (!CollectionUtils.isEmpty(configurers)) {
   this.configurers.addWebMvcConfigurers(configurers);
  }
 }
}

首先这个配置类其实就是我们上面最开始说的用继承来扩展功能的WebMvcConfigurationSupport,那么为什么实现WebMvcConfigurer接口也行?注意看上图中的代码,提供了一个configurers属性, 然后通过setConfigurers方法注入将ioc容器中的所有实现了WebMvcConfigurer接口的配置类都添加到configurers中;后续实现代码不是本章讨论范围,我也没有往下看,单看这里其实就已经明白了。

**上述是整合的基础,那么当我们在springboot中要添加功能的时候要注意一些什么事情呢?**这个也是写这篇文章的目的,因为最近在项目中有人在扩展功能的时候去继承了WebMvcConfigurationSupport这个类,然后联想到之前的项目也有人在springboot项目中使用了注解@EnableWebMvc,这两种情况都不会导致项目启动报错,但却在不该使用的时候使用了这些功能,导致了项目其实是不能正常使用的。现在来看一下为什么?

首先看一下springboot给我们提供的自动整合类,请参考类

org.springframework.boot.autoconfigure.web.servlet.WebMvcAutoConfiguration

我来截取部分代码

@Configuration
@ConditionalOnWebApplication(type = Type.SERVLET)
@ConditionalOnClass({ Servlet.class, DispatcherServlet.class, WebMvcConfigurer.class })
@ConditionalOnMissingBean(WebMvcConfigurationSupport.class)
@AutoConfigureOrder(Ordered.HIGHEST_PRECEDENCE + 10)
@AutoConfigureAfter({ DispatcherServletAutoConfiguration.class, TaskExecutionAutoConfiguration.class,
  ValidationAutoConfiguration.class })
public class WebMvcAutoConfiguration {    

    @Configuration
 @Import(EnableWebMvcConfiguration.class)
 @EnableConfigurationProperties({ WebMvcProperties.class, ResourceProperties.class })
 @Order(0)
 public static class WebMvcAutoConfigurationAdapter implements WebMvcConfigurer, ResourceLoaderAware {
    }
     /**
  * Configuration equivalent to {@code @EnableWebMvc}.
  */
 @Configuration
 public static class EnableWebMvcConfiguration extends DelegatingWebMvcConfiguration {
    }
}

现在说一下结论,springboot为我们提供的整合功能,已经默认的帮我们添加了很多功能,如消息转换器,静态资源映射,视图解析器,看下WebMvcAutoConfiguration的内部类WebMvcAutoConfigurationAdapter,其实就是实现了接口WebMvcConfigurer,然后再通过注解@Import(EnableWebMvcConfiguration.class)又将EnableWebMvcConfiguration这个配置类导入了进来,而我们点进去发现这个类的作用其实就是等同于之前我们说过的@EnableWebMvc。因此我们说的消息转换器啊,静态资源映射,视图解析器等这些默认实现就在WebMvcAutoConfiguration的内部类WebMvcAutoConfigurationAdapter又通过@Bean注入进来的

也就是说springboot其实帮我们整合好之后又默认帮我们做了一切常用的实现,这样我们开箱即用的不仅是整合好的框架,还有一些约定大于配置的功能,如静态资源要放在static下,其实就是默认帮我们做了资源映射,详细可以看下

org.springframework.boot.autoconfigure.web.servlet.WebMvcAutoConfiguration.WebMvcAutoConfigurationAdapter#addResourceHandlers

默认整合的功能基本满足于我们日常的开发,而如果我们还需要添加功能要怎么办呢?其实就是直接实现接口WebMvcConfigurer然后将当前类使用注解@Configuration标识为配置类即可。

那么为什么不能再继续继承接口WebMvcConfigurationSupport了呢?还是来看一下我们的自动配置类WebMvcAutoConfiguration吧,仔细看一下上面的配置类上的条件表达式中有这么一句非常非常重要的@ConditionalOnMissingBean(WebMvcConfigurationSupport.class),

上面我们所属的所有整合功能的前提是当前ioc容器中没有WebMvcConfigurationSupport这个bean,我们看到了springboot自己的实现是在当前自动配置类生效的时候才通过实现接口WebMvcConfigurer的,所以容器中在当前配置类未执行之前也是没有这个WebMvcConfigurationSupport的,现在我们突然在项目中添加功能的时候去继承了这个类,然后标识为配置类之后,立马在容器中就出现了这个bean,然后springboot就会以为我们要全面接管整合springmvc,我们要抛弃它的默认实现,然后自己玩。然后就悲剧了。现在整个mvc中反而只有我们自己新加的这个扩展空间了。这在绝大多数情况下根本不是我们想要的。

还有一个问题,为什么加注解@EnableWebMvc也不行?

其实通过上面的讲解我们已经能够看出来,这个注解其实就是导入了DelegatingWebMvcConfiguration这个配置类,而这个类就是继承WebMvcConfigurationSupport的,这两个效果是相同的,所以也不行。

总而言之一句话,在WebMvcAutoConfiguration这个配置类执行之前,无论是继承WebMvcConfigurationSupport还是在某个配置类上添加了注解@EnableWebMvc,都会导致容器中会被注入类型为WebMvcConfigurationSupport的bean,而springboot在实现自动配置时将这种行为定义成了你要自己去实现``

以上为个人经验,希望能给大家一个参考,也希望大家多多支持我们。

(0)

相关推荐

  • 在Spring Boot框架中使用AOP的正确姿势

    前言 Spring Boot是基于Spring的用来开发Web应用的框架,功能与Spring MVC有点类似,但是Spring Boot的一大特点就是需要的配置非常少.Spring Boot推荐convention over configuration,也就是约定大于配置,因此Spring Boot会帮你做许多自动的配置,并且Spring Boot使用的是Java Config,几乎可以做到零XML文件配置. 假设现在有这样一种场景,需要统计某个接口的处理耗时,我们可以使用AOP来实现,AOP为

  • springboot与springmvc基础入门讲解

    目录 一,SpringBoot –1,概述 –2,用法 二,SpringMVC –1,概述 –2,原理 –3,入门案例 总结 一,SpringBoot –1,概述 用来整合maven项目,可以和Spring框架无缝衔接. –2,用法 –1,创建SpringBoot工程:File-New-Project-选择Spring Init--next-输入groupId.项目id.选成jdk8-next-选择SpringWeb-ok –2,配置Maven:File-Settings-选择Build--Ma

  • SpringBoot集成SpringMVC的方法示例

    Spring MVC是一款优秀的.基于MVC思想的应用框架,它是Spring的一个子框架.是当前最优秀的MVC框架. Spring Boot整合Spring MVC只需在pom.xml中引入 <dependency> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-starter-web</artifactId> <version>2.3.7.RE

  • springboot扩展MVC的方法

    springboot扩展MVC 自定义 config -> SpringMvcConfig.java 下边就是扩展springMVC的模板: 第一步:@Configuration 注解的作用:让这个类变为配置类. 第二步:必须实现 WebMvcConfigurer 接口. 第三步:重写对应的方法. package com.lxc.springboot.config; import org.springframework.context.annotation.Configuration; impo

  • springboot+springmvc+mybatis项目整合

    介绍: 上篇给大家介绍了ssm多模块项目的搭建,在搭建过程中spring整合springmvc和mybatis时会有很多的东西需要我们进行配置,这样不仅浪费了时间,也比较容易出错,由于这样问题的产生,Pivotal团队提供了一款全新的框架,该框架使用了特定的方式来进行配置,从而使开发人员不再需要定义样板化的配置.通过这种方式,Spring Boot致力于在蓬勃发展的快速应用开发领域(rapid application development)成为领导者. 特点: 1. 创建独立的Spring应用

  • 在springboot中添加mvc功能的正确姿势讲解

    springboot 添加mvc功能 先放出来几个类(包含注解或接口)来观摩一下 WebMvcConfigurer @EnableWebMvc WebMvcConfigurerAdapter(已过时,不再详述,可以理解为继承该类有和实现WebMvcConfigurer一样的效果) WebMvcConfigurationSupport WebApplicationInitializer 这里只聊springboot或者无web.xml环境的情况,无论如何得看一下这个祖宗,以下代码来源于spring

  • 详解SpringBoot中添加@ResponseBody注解会发生什么

    SpringBoot版本2.2.4.RELEASE. [1]SpringBoot接收到请求 ① springboot接收到一个请求返回json格式的列表,方法参数为JSONObject 格式,使用了注解@RequestBody 为什么这里要说明返回格式.方法参数.参数注解?因为方法参数与参数注解会影响你使用不同的参数解析器与后置处理器!通常使用WebDataBinder进行参数数据绑定结果也不同. 将要调用的目标方法如下: @ApiOperation(value="分页查询") @Re

  • jsp实现仿QQ空间新建多个相册名称并向相册中添加照片功能

    工具:Eclipse,Oracle,smartupload.jar:语言:jsp,Java:数据存储:Oracle. 实现功能介绍: 主要是新建相册,可以建多个相册,在相册中添加多张照片,删除照片,删除相册,当相册下有照片时先删除照片才能删除相册. 因为每个相册和照片要有所属人,所以顺带有登录功能. 声明:只是后端实现代码,前台无任何样式,代码测试可行,仅供参考. 代码: 数据库连接帮助类: public class JDBCHelper { public static final String

  • 在SpringBoot中添加Redis及配置方法

    在实际的开发中,会有这样的场景.有一个微服务需要提供一个查询的服务,但是需要查询的数据库表的数据量十分庞大,查询所需要的时间很长. 此时就可以考虑在项目中加入缓存. 引入依赖 在maven项目中引入如下依赖.并且需要在本地安装redis. <dependency> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-starter-data-redis</artifac

  • 如何在django中添加日志功能

    官方文档 猛戳这里 在settings中配置以下代码 #LOGGING_DIR 日志文件存放目录 LOGGING_DIR = "logs" # 日志存放路径 if not os.path.exists(LOGGING_DIR): os.mkdir(LOGGING_DIR) import logging LOGGING = { 'version': 1, 'disable_existing_loggers': False, 'formatters': { #格式化器 'standard'

  • JS一分钟在github+Jekyll的博客中添加访问量功能的实现

    目录 一分钟在github+Jekyll博客中添加访问量功能 一.Jekll是什么 1. Jekll是软件简介 二.不蒜子 1. 不蒜子简介: 2. 官网入口 3. 使用方法 4. 个性化 总结 一分钟在github+Jekyll博客中添加访问量功能 前言 不会还有小伙伴不知道,github+Jekll是可以搭建自己的个人博客吧??? 如果大家觉得csdn的blog不够高大上,或者私密性不好,不能抒发一些情感,那么可以搭一个github的博客,只需一小时即可,详见大佬的博文: 一小时搭建gith

  • springboot切面添加日志功能实例详解

    1.新建一个springboot项目 2.定义个切面类,并指定切入点,获取所需记录信息(如:访问人IP, 访问地址,访问地址名称等) 3.新建数据库 SET FOREIGN_KEY_CHECKS=0; -- ---------------------------- -- Table structure for ds_access_log -- ---------------------------- DROP TABLE IF EXISTS `ds_access_log`; CREATE TAB

  • 浅谈Go中数字转换字符串的正确姿势

    在许多语言中,你可以轻松地将任何数据类型转换为字符串,只需将其与字符串连接,或者使用类型转换表达式即可.但是,如果你在Go中尝试执行似乎很明显的操作(例如将int转换为字符串),你不太可能获得预期的效果. 比如下面: string(120) 你期望返回的结果是什么?如果你有使用其他编程语言的经验,那么大多数人的猜测是" 123".相反,在Go中上面的类型转换会得到" E"之类的值,那根本不是我们想要的.因为string()会直接把字节或者数字转换为字符的UTF-8表

  • Android实现动态向Gallery中添加图片及倒影与3D效果示例

    本文实例讲述了Android实现动态向Gallery中添加图片及倒影与3D效果的方法.分享给大家供大家参考,具体如下: 在Android中gallery可以提供一个很好的显示图片的方式,实现上面的效果以及动态添加数据库或者网络上下载下来的图片资源.我们首先实现一个自定义的Gallery类. MyGallery.java: package nate.android.Service; import android.content.Context; import android.graphics.Ca

  • 给Python的Django框架下搭建的BLOG添加RSS功能的教程

    前些天有位网友建议我在博客中添加RSS订阅功能,觉得挺好,所以自己抽空看了一下如何在Django中添加RSS功能,发现使用Django中的syndication feed framework很容易实现. 具体实现步骤和代码如下:     1.Feed类 # -*- coding: utf-8 -*- from django.conf import settings from django.contrib.syndication.views import Feed from django.util

随机推荐