SpringBoot之MVC配置(WebMvcConfigurer详解)

偏执的太偏执、 2024-03-17 18:25 146阅读 0赞

一:基本介绍

  Spring MVC是一种常用的Java Web框架,它提供了一种基于MVC模式的开发方式,可以方便地实现Web应用程序。在Spring MVC中,WebMvcConfigurer是一种常用的配置方式,可以允许我们自定义Spring MVC的行为,比如添加拦截器、消息转换器等。在本文中,我们将介绍什么是WebMvcConfigurer,以及如何使用它来自定义Spring MVC的配置。可以看到WebMvcConfigurer是一个非常灵活和强大的工具,它可以让我们实现自己的业务需求并提高代码的可读性和可维护性。而且我们在Spring、SpringBoot都可以很简单的使用WebMvcConfigurer,下面主要在SpringBoot中说明配置

1:提示说明

  其实在Spring Boot 1.5版本都是靠重写WebMvcConfigurerAdapter的方法来添加自定义拦截器,消息转换器等。但是到了SpringBoot 2.0之后,WebMvcConfigurerAdapter被标记为@Deprecated(弃用)。官方推荐直接实现WebMvcConfigurer或者直接继承WebMvcConfigurationSupport,这两种区别如下:(具体参考官方API:6.0.9

c1b208da1f20334fce3953396cd9caa9.gif

  1. WebMvcConfigurerWebMvcConfigurationSupport都是Spring MVC中的组件,它们都可以用于配置Spring MVC的一些特性。
  2. 具体的区别如下:
  3. ①:实现方式不同:
  4. WebMvcConfigurer
  5. 是一个接口,它提供了多个回调方法,可以用于自定义Spring MVC的配置(如消息转换器、拦截器等)。我们在使用时只需要实现
  6. 该接口,重写其中的方法即可。
  7. WebMvcConfigurationSupport
  8. 是一个抽象类,它也提供了多个回调方法,用于自定义Spring MVC的配置,但是需要继承该类并重写其中的方法。
  9. ②:作用不同:
  10. WebMvcConfigurer
  11. 主要用于添加或修改Spring MVC的配置,如添加拦截器,自定义消息转换器等。
  12. WebMvcConfigurationSupport
  13. 主要用于完全自定义Spring MVC的配置,如果我们需要对Spring MVC的配置进行大量的自定义,可以选择继承该类并重写其中的
  14. 方法。但是需要注意的是,继承该类会覆盖Spring MVC的部分默认配置。
  15. 因此,当我们只需要对部分配置进行自定义时,应该使用WebMvcConfigurer
  16. ③:继承关系不同
  17. WebMvcConfigurer
  18. 没有继承关系,我们只需要实现该接口即可使用。
  19. WebMvcConfigurationSupport
  20. 是一个抽象类,需要继承后才能使用。
  21. 总结:在日常开发中推荐优先使用WebMvcConfigurer的方式,因为简单方便,也没有特别复杂的定制需求;
  22. 若我们项目中使用的MVC存在着更加复杂的配置需求推荐WebMvcConfigurationSupport,通过继承此类,我们可以说对官方的MVC代码进行
  23. 重写操作,但是因为其配置量较大,实现比较复杂,因此在日常开发中使用WebMvcConfigurationSupport并不常见。

b467d55695c637c172834c57ef86790c.gif

2:MVC配置简要

4594744f90c5b2173b301686c71f4cfd.gif

  1. MVC配置,其实就是说在WebMvcConfigurer接口提供了很多种自定义配置,需要我们自定义配置,其常用配置如下:
  2. 1addInterceptors(拦截器配置)
  3. 这个方法可用于配置拦截器。
  4. 2addCorsMappings(全局跨域处理)
  5. 这个方法用来配置跨域访问的规则。
  6. 3addViewControllers(注册视图控制器)
  7. 这个方法可以注册一个或多个视图控制器,让我们写的地址可以对应一个资源文件,如html文件
  8. 4addResourceHandlers(配置静态资源处理)
  9. 方法可用于配置静态资源处理器。可以在客户端直接访问静态资源信息

43403af754b1b5b90e609495770f73ea.gif

3:Spring和SpringBoot配置WebMvcConfigurer区别

a6498345eebf36f2c8d9ac3706bee5bd.gif

  1. Spring中配置WebMvcConfigurer方式:
  2. ①:创建一个 Java 类,并实现WebMvcConfigurer接口
  3. @Configuration
  4. public class MyWebMvcConfig implements WebMvcConfigurer {
  5. // 自定义配置代码
  6. }
  7. ②:注入到Bean容器里
  8. @Configuration
  9. public class AppConfig {
  10. @Bean
  11. public MyWebMvcConfig myWebMvcConfig() {
  12. return new MyWebMvcConfig();
  13. }
  14. }
  15. SpringBoot中配置WebMvcConfigurer方式:(这种方式简单)
  16. ①:创建一个 Java 类,并实现WebMvcConfigurer接口就可以了
  17. @Configuration
  18. public class MyWebMvcConfig implements WebMvcConfigurer {
  19. // 自定义配置代码
  20. }

1977c9bf1531ff79fff09bad034ecb6c.gif

回到目录 ↑↑↑

二:拦截器配置(addInterceptors)

  在SpringBoot中,我们可以使用拦截器来对请求进行统一的预处理或后处理。拦截器可以用于日志记录、权限检查、性能监控、事务控制等方面,是一个非常重要的组件。要在SpringBoot中实现拦截器,则首先要创建一个实现HandlerInterceptor接口的拦截器类。该接口定义了三个方法,分别是preHandle、postHandle和afterCompletion,用于在请求处理前、请求处理后和请求完成后进行处理。

786479aadd06daa912bfd48e6a05a35f.gif

  1. HandlerInterceptor接口方法详解:
  2. ①:preHandler
  3. 在请求处理之前被调用。该方法在Interceptor类中最先执行,用来进行一些前置初始化操作或是对当前请求做预处理,
  4. 也可以进行一些判断来决定请求是否要继续进行下去。该方法的返回值是Boolean类型,当它返回false时,表示请求结束,
  5. 后续的InterceptorController都不会再执行;当它返回为true时会继续调用下一个InterceptorpreHandle方法,
  6. 如果已经是最后一个Interceptor的时候就会调用当前请求的Controller方法。
  7. ②:postHandler
  8. 在请求处理完成之后调用。也就是Controller方法调用之后执行,但是它会在DispatcherServlet进行视图返回渲染之前被调用,
  9. 所以我们可以在这个方法中对Controller处理之后的ModelAndView对象进行操作。
  10. ③:afterCompletion
  11. 在整个请求结束后调用。就是对应的Interceptor类的postHandler方法返回true时才执行。就是说该方法将在整个请求结束之后,
  12. 也就是在DispatcherServlet渲染了对应的视图之后执行。此方法主要用来进行资源清理。
  13. 注:官方其实不建议我们非要把3个方法都重写,我们只要对需要的方法重写接口,就比如大部分项目只需要重写preHandler方法

470831e0666682136ddb716dbd20c3b2.gif

00ac397d66769e68e7b7217875f0ac0f.gif

示例代码 实现拦截器接口(用于配置拦截器)

我们配置完拦截器之后,这时拦截器只是一个未被注册的普通类,这时需要把一个或者多个拦截器注册到WebMvcConfigurer

526f92e70b9467f8dd58662541e772ee.gif

  1. InterceptorRegistry类方法介绍:
  2. ①:addInterceptor
  3. 该方法用于向拦截器链中添加一个拦截器。interceptor参数为待添加的拦截器对象,可以是自定义的拦截器类或Spring提供的预置
  4. 拦截器。返回值为InterceptorRegistration对象,用于进一步配置该拦截器的属性。
  5. ②:addWebRequestInterceptor
  6. 该方法用于向WebRequest拦截器链中添加一个拦截器。interceptor参数为待添加的拦截器对象,
  7. 可以是自定义的WebRequestInterceptor类或者Spring提供的预置拦截器。
  8. 也是返回值为InterceptorRegistration对象,用于进一步配置该拦截器的属性。
  9. ③:getInterceptors
  10. 用于获取当前已经添加到拦截器链中的所有拦截器,返回值为List<HandlerInterceptor>对象,表示拦截器列表。
  11. InterceptorRegistration类方法介绍:
  12. ①:order
  13. 该方法用于设置拦截器的执行顺序,即在拦截器链中的位置。order参数为一个整数,值越小表示越先执行。
  14. ②:addPathPatterns
  15. 该方法用于设置需要拦截的请求路径模式,即满足哪些请求路径时才会触发该拦截器。若"/**"则拦截全部;
  16. 传入的参数是一个字符串数组,包含多个Ant风格的路径模式,例如 "/api/**""/user/*"等。
  17. ③:excludePathPatterns
  18. 该方法用于设置不需要拦截的请求路径模式,即满足哪些请求路径时不会触发该拦截器。一般不拦截,如登录或者Swagger
  19. 传入的参数是一个字符串数组,包含多个Ant风格的路径模式,例如 "/api/login""/user/login"等。
  20. ④:pathMatcher
  21. 该方法用于设置该拦截器所使用的PathMatcher实例,从而可以自定义路径匹配逻辑。

796044606d628a1d2330a507972744aa.gif

01b17fa285eaa1935d3785971fdeab92.gif

示例代码 在WebMvcConfigurer类上注册创建的拦截器

  注:拦截的路径或者放行的路径是以Controller开始的,如我们在application.yml配置的地址前缀则不包含

回到目录 ↑↑↑

三:跨域配置(addCorsMappings)

  如何是跨域,大家应该都了解,这里我将在WebMvcConfigurer中解决跨域问题,其实跨域的解决方式很多,这里我就简单说明,具体参考:SpringBoot如何解决跨域的三种方式

07e1a8f3ce355ab8b78d5746bcae78a2.gif

  1. CorsRegistry类方法介绍:
  2. ①:addMapping
  3. 该方法用于添加允许跨域访问的路径,String类型,若存在多个路径则需要在CorsRegistry里配置多个
  4. CorsRegistration类方法介绍:
  5. CorsRegistrationCorsRegistry的辅助类,使用它可以对单个跨域请求进行更细粒度的配置。
  6. ①:allowedOrigins(低版本使用,但是现在高版本也支持)
  7. 设置允许跨域请求的来源URL。该方法接受多个参数,每个参数为一个允许的来源URL。或者设置"*"
  8. ②:allowedOriginPatterns(一般使用这种方式)
  9. 设置允许跨域请求的来源URL的模式。该方法接受多个参数,每个参数为一个允许的来源URL模式。或者设置"*"
  10. ③:allowCredentials
  11. 设置是否允许跨域请求携带凭证信息。默认情况下,浏览器不会向跨域请求发送Cookie等凭证信息。
  12. 如果希望携带凭证信息,则需要将allowCredentials方法设置为true
  13. ④:allowedMethods
  14. 设置允许跨域请求的HTTP方法。该方法接受多个参数,每个参数为一种允许的HTTP请求方式。
  15. ⑤:allowedHeaders
  16. 设置允许请求携带的HTTP头信息。该方法接受多个参数,每个参数为一种允许的HTTP头信息。(放行哪些请求头部信息)
  17. ⑥:exposedHeaders
  18. 设置响应头信息,这些信息允许客户端访问。该方法接受多个参数,每个参数为一种允许的响应头信息。(暴露哪些响应头信息)
  19. ⑦:combine
  20. 将当前CorsRegistration对象与另一个CorsConfiguration对象合并,返回合并后的CorsConfiguration对象。
  21. 可以使用该方法将多个CorsRegistration对象的配置合并到同一个CorsConfiguration对象中。
  22. ⑧:maxAge
  23. 设置响应的缓存时间,单位为秒,默认30分钟。
  24. 例如,当设置maxAge3600时,如果浏览器在一小时内再次向同一个目标URL发送跨域请求,
  25. 就可以直接使用以前的预检请求结果,而不需要再次进行预检请求。maxAge属性只影响预检请求的缓存时间,
  26. 而不会影响正常的跨域请求,因此不会对实际的业务逻辑产生影响。此外,maxAge属性的具体值需要根据实际情况进行调整,
  27. 过小的缓存时间可能会导致频繁的预检请求,过大的缓存时间可能会使跨域请求的控制权得不到及时更新,从而增加安全风险。

a6498345eebf36f2c8d9ac3706bee5bd.gif

d463849b400b14e9439fdd95d8b6c7f9.gif

测试跨域请求发送(直接启动这个html,配置可能跨域的地址即可)

d8797d71fc15380c1d14df9bc3064de6.gif

示例代码 跨域放行代码配置

回到目录 ↑↑↑

四:注册页面跳转(addViewControllers)

  addViewControllers方法是SpringMVC中WebMvcConfigurer接口定义的一个方法,用于注册一个简单的视图控制器,以便将请求路径映射到一个具体的视图页面。在一些简单的场景下,我们可能只需要将某个请求路径直接映射到一个固定的视图页面,而不需要进行额外的逻辑处理。此时,可以使用addViewControllers方法来快速注册一个视图控制器,并指定对应的请求路径和视图名称即可。但是我们需要额外注意的是,我们在SpringBoot中处理视图跳转时最好集成Thymeleaf,因为它是SpringBoot指定认可的并且,Thymeleaf与SpringMVC协同工作,可以方便地实现快速开发Web应用程序。

738e71147a857492bc935f0fbba0769d.gif

  1. SpringBoot配置Thymeleaf的一些信息:
  2. 导入Thymeleaf坐标:
  3. <!--导入SpringBoot集成Thymeleaf启动器-->
  4. <dependency>
  5. <groupId>org.springframework.boot</groupId>
  6. <artifactId>spring-boot-starter-thymeleaf</artifactId>
  7. </dependency>
  8. application.yml配置Thymeleaf配置信息:
  9. spring:
  10. # 配置Thymeleaf模板(默认启动会请求/templates/index.html)
  11. thymeleaf:
  12. cache: false # 是否有模板缓存
  13. prefix: classpath:/templates/ # 模板放置的位置
  14. suffix: .html # 模板后缀
  15. mode: HTML # 模板类型
  16. encoding: UTF-8 # 模板编码

defd8e974a67b266e0684e05d65019c9.gif

  配置完基本的信息后我们就可以进行资源视图的跳转了,如下一些基本介绍:

3df0a6906ec3fc51de5c3f9bad12a54a.gif

  1. ViewControllerRegistry类说明:
  2. ①:addViewController(String urlPath)
  3. 通过urlPath参数指定的请求URL路径(例如"/home")注册一个简单的视图控制器,
  4. 该方法返回一个ViewControllerRegistration对象,通过该对象可以设置相关属性,如视图名称、请求方式等。
  5. 如:registry.addViewController("/login");
  6. ②:setOrder(int order)
  7. 设置当前视图控制器的执行顺序,当有多个视图控制器针对同一请求路径时,可以使用该方法进行优先级排序。
  8. 默认情况下,不同的视图控制器按照它们被注册的顺序执行。
  9. ③:addRedirectViewController(String urlPath, String redirectUrl)
  10. 注册一个重定向视图控制器,将urlPath请求路径重定向到指定的重定向地址redirectUrl
  11. 如:registry.addRedirectViewController("/toBaidu","https://www.baidu.com");
  12. ViewControllerRegistration类说明:
  13.   ①:setViewName(String viewName)
  14. 资源路径的前缀
  15.   ②:setStatusCode(HttpStatus statusCode)
  16. 配置访问不存在资源的响应码,如下常见的:
  17. HttpStatus.BAD_REQUEST:请求参数错误或格式不正确,例如缺少必需参数、参数类型错误等。
  18. HttpStatus.UNAUTHORIZED:未经授权访问,需要用户登录或提供凭证。
  19. HttpStatus.FORBIDDEN:已经授权但访问被禁止,通常意味着权限不足或需要进行进一步身份验证。
  20. HttpStatus.NOT_FOUND:请求的资源不存在,通常使用自定义的404错误页面进行提示。
  21. HttpStatus.METHOD_NOT_ALLOWED:请求方式不支持,例如GET请求访问只支持POST的接口时会返回405错误。
  22. HttpStatus.INTERNAL_SERVER_ERROR:服务器内部错误,需要在后台进行排查和修复。
  23. 示例:registry.addViewController("/**").setStatusCode(HttpStatus.NOT_FOUND);
  24. 说明:访问不存在的页面我一律按照404处理,但是我们templates/error/404.html页面需要存在
  25. 说明:按照我们之前配置的thymeleaf配置来说,默认根路径为resources/templates,跳转的资源文件都是.html文件

4608bb9785937dd11a27b849206b3688.gif

324cf0f78f78ba7d139c435c563338bd.gif

示例代码 注册页面跳转配置

回到目录 ↑↑↑

五:配置静态资源处理(addResourceHandlers)

  addResourceHandlers方法是SpringMVC框架提供的一种配置静态资源的方式,它可以将指定的资源路径映射到一个或多个URL路径上,并指定资源的缓存策略、版本号以及是否允许目录列表等选项。具体来说,addResourceHandlers方法需要传入一个ResourceHandlerRegistry对象作为参数,然后在这个对象上调用addResourceHandler方法,来添加一个或多个处理器。

65ba996606634d47a6d771fc9537dc53.gif

  1. ResourceHandlerRegistry类方法介绍:
  2. ①:addResourceHandler
  3. 该方法用于指定静态资源的URL路径,支持Ant风格的通配符,如“/resources/**”表示匹配所有以“/resources/”开头的请求。
  4. ResourceHandlerRegistration类方法介绍:
  5. ①:addResourceLocations
  6. 该方法为静态资源所在的物理路径或URL。可以使用多个addResourceLocations方法指定多个路径,如下例:
  7. registry.addResourceHandler("/resources/**")
  8. .addResourceLocations("classpath:/static/", "file:/opt/files/")
  9. 说明:
  10. classpath:/static/表示在项目的Classpath下(即src/main/resources文件夹下)查找static文件夹,
  11. file:/opt/files/表示在系统中的/opt/files/目录下查找文件。
  12. ②:setCacheControl
  13. 此方法用于设置缓存控制头(cache-control header),CacheControl是一个封装了缓存策略的类。例如:
  14. CacheControl cc = CacheControl.maxAge(30, TimeUnit.DAYS).cachePublic();
  15. registry.addResourceHandler("/resources/**").addResourceLocations("classpath:/static/")
  16. .setCacheControl(cc);
  17. 说明:这将指示浏览器缓存静态资源30天,并且它们是public缓存,意味着中间代理服务器也可以缓存资源。
  18. ③:setCachePeriod
  19. 该方法用于设置静态资源缓存时间,参数类型为Duration类型。如:
  20. registry.addResourceHandler("/resources/**").addResourceLocations("classpath:/static/")
  21. .setCachePeriod(Duration.ofMinutes(5));
  22. 说明:静态资源缓存的过期时间为5分钟
  23. ④:setOptimizeLocations
  24. 此方法用于启用或禁用位置优化。如果启用位置优化,则将优化静态资源的位置,以便并发访问静态资源时可以获得更好的性能。
  25. 默认情况下,位置优化是禁用的。
  26. ⑤:setUseLastModified
  27. 此方法用于启用或禁用上次修改时间检查(last-modified check)。如果启用上次修改时间检查,则在每个请求中发送一
  28. 个if-modified-since头,以检查是否需要返回新内容。默认情况下,上次修改时间检查是启用的。
  29. ⑥:resourceChain
  30. 用于开启或关闭ResourceChain模式。当开启ResourceChain模式时,每个资源文件都会自动添加版本号,避免浏览器缓存问题。
  31. 例如:registry.addResourceHandler("/resources/**").addResourceLocations("classpath:/static/")
  32. .resourceChain(true);
  33. 请注意,要使ResourceChain生效,还需要设置addResolver
  34. 如:.resourceChain(false)
  35. // 添加VersionResourceResolver,且指定版本号
  36. .addResolver(new VersionResourceResolver()
  37. .addFixedVersionStrategy("1.0.0", "/**"));
  38. 下次访问地址:http://localhost:8881/resources/1.0.0/xxx.css

d62737287f3c35cff50f788c043b9261.gif

ae690e1f2dfeb87f70e7f69d12f75ddf.gif

示例代码 配置静态资源处理

其实还有许许多多的配置信息,后面会慢慢补充。

发表评论

表情:
评论列表 (有 0 条评论,146人围观)

还没有评论,来说两句吧...

相关阅读