WebMvcConfigurer
- 一、页面跳转控制器
- step1:创建视图,resources/templates/index.html
- step2:创建SpringMVC配置类
- step3:测试功能
- 二、数据格式化
- step1:创建 DeviceInfo 数据类
- step2:自定义 Formatter
- step3: 登记自定义的 DeviceFormatter
- step4: 新建 Controller 接受请求设备数据
- step5:单元测试
- 三、拦截器
- (一)、一个拦截器
- step1:创建文章的 Controller
- step2:创建有关权限拦截器
- step3:登记拦截器
- step4:测试拦截器
- (二)、多个拦截器
- step1:创建登录拦截器
- step2:登记拦截器,设置顺序 order
- step3:测试拦截器
WebMvcConfigurer 作为配置类是,采用 JavaBean 的形式来代替传统的 xml 配置文件形式进行针对框架个性化定制,就是 Spring MVC XML 配置文件的 JavaConfig(编码)实现方式。自定义 InterceptorViewResolver,
MessageConverter。WebMvcConfigurer 就是 JavaConfig 形式的 Spring MVC 的配置文件WebMvcConfigurer 是一个接口,需要自定义某个对象,实现接口并覆盖某个方法。主要方法功能介绍一下:
public interface WebMvcConfigurer {
//帮助配置 HandlerMapping
default void configurePathMatch(PathMatchConfigurer configurer) {
}
//处理内容协商
default void configureContentNegotiation(ContentNegotiationConfigurer configurer) {
}
//异步请求
default void configureAsyncSupport(AsyncSupportConfigurer configurer) {
}
//配置默认 servlet
default void configureDefaultServletHandling(DefaultServletHandlerConfigurer configurer) {
}
//配置内容转换器
default void addFormatters(FormatterRegistry registry) {
}
//配置拦截器
default void addInterceptors(InterceptorRegistry registry) {
}
//处理静态资源
default void addResourceHandlers(ResourceHandlerRegistry registry) {
}
//配置全局跨域
default void addCorsMappings(CorsRegistry registry) {
}
//配置视图页面跳转
default void addViewControllers(ViewControllerRegistry registry) {
}
//配置视图解析器
default void configureViewResolvers(ViewResolverRegistry registry) {
}
//自定义参数解析器,处理请求参数
default void addArgumentResolvers(List<HandlerMethodArgumentResolver> resolvers) {
}
//自定义控制器方法返回值处理器
default void addReturnValueHandlers(List<HandlerMethodReturnValueHandler> handlers) {
}
//配置 HttpMessageConverters
default void configureMessageConverters(List<HttpMessageConverter<?>> converters) {
}
//配置 HttpMessageConverters
default void extendMessageConverters(List<HttpMessageConverter<?>> converters) {
}
//配置异常处理器
default void configureHandlerExceptionResolvers(List<HandlerExceptionResolver> resolvers) {
}
//扩展异常处理器
default void extendHandlerExceptionResolvers(List<HandlerExceptionResolver> resolvers) {
}
//JSR303 的自定义验证器
default Validator getValidator() {
return null;
}
//消息处理对象
default MessageCodesResolver getMessageCodesResolver() {
return null;
}
}
一、页面跳转控制器
Spring Boot 中使用页面视图,比如 Thymeleaf。要跳转显示某个面,必须通过 Controller 对象。也就是我们需要创建一个 Controller,转发到一个视图才行。 如果我们现在需要显示页面,可以无需这个 Controller。
addViewControllers() 完成从请求到视图跳转。
- 需求:访问/welcome 跳转到项目首页 index.html(Thyemeleaf 创建的对象)
项目代码结构:
step1:创建视图,resources/templates/index.html
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8">
<title>Title</title>
</head>
<body>
<h3>欢迎各位小伙伴!!!</h3>
</body>
</html>
step2:创建SpringMVC配置类
- 重写addViewControllers()方法
- 配置页面控制:addViewController(“请求的uri”).指定他的视图setViewName(目标视图)
package com.bjpowernode.mvc.settings;
import org.springframework.context.annotation.Configuration;
import org.springframework.web.servlet.config.annotation.ViewControllerRegistry;
import org.springframework.web.servlet.config.annotation.WebMvcConfigurer;
/**
* springmvc配置类
*/
@Configuration
public class MvcSettings implements WebMvcConfigurer {
//页面跳转控制器,从请求直达视图
@Override
public void addViewControllers(ViewControllerRegistry registry) {
//配置页面控制:addViewController("请求的uri").指定他的视图setViewName(目标视图)
registry.addViewController("/welcome").setViewName("index");
}
}
step3:测试功能
浏览器访问http://localhost:8080/welcome
二、数据格式化
Formatter<T>是数据转换接口,将一种数据类型转换为另一种数据类型。与 Formatter<T>功能类型的还有Converter<S,T>。本节研究 Formatter<T>接口。Formatter<T>只能将 String 类型转为其他数据数据类型。这点在Web 应用适用更广。因为 Web 请求的所有参数都是 String,我们需要把 String 转为 Integer ,Long,Date 等等。
Spring 中内置了一下 Formatter:
- DateFormatter : String 和 Date 之间的解析与格式化
- InetAddressFormatter :String 和 InetAddress 之间的解析与格式化
- PercentStyleFormatter :String 和 Number 之间的解析与格式化,带货币符合
- NumberFormat :String 和 Number 之间的解析与格式化
- 我在使用@ DateTimeFormat , @NumberFormat 注解时,就是通过 Formatter解析 String 类型到我们期望的Date 或 Number 类型
- Formatter也是 Spring 的扩展点,我们处理特殊格式的请求数据时,能够自定义合适的 Formatter,将请求的 String 数据转为我们的某个对象,使用这个对象更方便我们的后续编码
接口原型
Formatter是一个组合接口,没有自己的方法。内容来自 Printer和 Parser
- Printer:将 T 类型转为 String,格式化输出
- Parser:将 String 类型转为期望的 T 对象。
- 我们项目开发,可能面对多种类型的项目,复杂程度有简单,有难一些。特别是与硬件打交道的项目,数据的格式与一般的 name: lisi, age:20 不同。数据可能是一串“1111; 2222; 333,NF; 4; 561” 。
- 需求:将“1111;2222;333,NF;4;561”接受,代码中用 DeviceInfo 存储参数值
step1:创建 DeviceInfo 数据类
import lombok.Data;
@Data
public class DeviceInfo {
private String item1;
private String item2;
private String item3;
private String item4;
private String item5;
}
step2:自定义 Formatter
package com.bjpowernode.mvc.fomatter;
import com.bjpowernode.mvc.model.DeviceInfo;
import org.springframework.format.Formatter;
import org.springframework.util.StringUtils;
import java.text.ParseException;
import java.util.Locale;
import java.util.StringJoiner;
/**
* 将请求参数字符串转为 DeviceInfo
*/
public class DeviceFormatter implements Formatter<DeviceInfo> {
@Override
public DeviceInfo parse(String text, Locale locale) throws ParseException {
DeviceInfo info = null;
if(StringUtils.hasLength(text)){
String[] items = text.split(";");
info = new DeviceInfo();
info.setItem1(items[0]);
info.setItem2(items[1]);
info.setItem3(items[2]);
info.setItem4(items[3]);
info.setItem5(items[4]);
}
return info;
}
@Override
public String print(DeviceInfo object, Locale locale) {
StringJoiner joiner = new StringJoiner("#");
joiner.add(object.getItem1()).add(object.getItem2())
.add(object.getItem3()).add(object.getItem4()).add(object.getItem5());
return joiner.toString();
}
}
step3: 登记自定义的 DeviceFormatter
- addFormatters() 方法登记 Formatter
package com.bjpowernode.mvc.settings;
import com.bjpowernode.mvc.fomatter.DeviceFormatter;
import org.springframework.context.annotation.Configuration;
import org.springframework.format.FormatterRegistry;
import org.springframework.web.servlet.config.annotation.ViewControllerRegistry;
import org.springframework.web.servlet.config.annotation.WebMvcConfigurer;
/**
* springmvc配置类
*/
@Configuration
public class MvcSettings implements WebMvcConfigurer {
//页面跳转控制器,从请求直达视图
@Override
public void addViewControllers(ViewControllerRegistry registry) {
//配置页面控制:addViewController("请求的uri").指定他的视图setViewName(目标视图)
registry.addViewController("/welcome").setViewName("index");
}
@Override
public void addFormatters(FormatterRegistry registry) {
registry.addFormatter(new DeviceFormatter());
}
}
step4: 新建 Controller 接受请求设备数据
import com.bjpowernode.mvc.model.DeviceInfo;
import org.springframework.web.bind.annotation.PostMapping;
import org.springframework.web.bind.annotation.RequestParam;
import org.springframework.web.bind.annotation.RestController;
@RestController
public class DeviceController {
@PostMapping("/device/add")
public String addDeviceInfo(@RequestParam("device")DeviceInfo info){
return "接受设备信息"+info.toString();
}
}
step5:单元测试
POST http://localhost:8080/device/add
Content-Type: application/x-www-form-urlencoded
device=1111;2222;333,NF;4;561
三、拦截器
HandlerInterceptor 接口和它的实现类称为拦截器,是 SpringMVC 的一种对象。拦截器是 Spring MVC 框架的对象与 Servlet 无关。拦截器能够预先处理发给 Controller 的请求。可以决定请求是否被 Controller 处理。用户请求是先由 DispatcherServlet 接收后,在 Controller 之前执行的拦截器对象。
一个项目中有众多的拦截器:框架中预定义的拦截器, 自定义拦截器。下面我说说自定义拦截器的应用。根据拦截器的特点,类似权限验证,记录日志,过滤字符,登录 token 处理都可以使用拦截器。
拦截器定义步骤:
- 声明类实现 HandlerInterceptor 接口,重写三个方法(需要那个重写那个)
- 登记拦截器
(一)、一个拦截器
需求:zhangsan 操作员用户,只能查看文章,不能修改,删除。
step1:创建文章的 Controller
import org.springframework.web.bind.annotation.DeleteMapping;
import org.springframework.web.bind.annotation.GetMapping;
import org.springframework.web.bind.annotation.PostMapping;
import org.springframework.web.bind.annotation.RestController;
@RestController
public class ArticleController {
@PostMapping("/article/add")
public String addArticle(){
return "发布新的文章";
}
@PostMapping("/article/edit")
public String editArticle(){
return "修改文章";
}
@DeleteMapping("/article/remove")
public String removeArticle(){
return "删除文章";
}
@GetMapping("/article/query")
public String query(){
return "查询文章";
}
}
step2:创建有关权限拦截器
import jakarta.servlet.http.HttpServletRequest;
import jakarta.servlet.http.HttpServletResponse;
import org.springframework.web.servlet.HandlerInterceptor;
public class AuthInterceptor implements HandlerInterceptor {
private static final String COMMON_USER="zhangsan";
@Override
public boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler) throws Exception {
System.out.println("====AuthInterceptor权限拦截器====");
String loginUser = request.getParameter("loginUser");
String requestURI = request.getRequestURI();
if(COMMON_USER.equals(loginUser) && (
requestURI.startsWith("/article/add")||
requestURI.startsWith("/article/edit")||
requestURI.startsWith("/article/remove")
)){
return false;
}
return true;
}
}
step3:登记拦截器
package com.bjpowernode.mvc.settings;
import com.bjpowernode.mvc.fomatter.DeviceFormatter;
import com.bjpowernode.mvc.interceptor.AuthInterceptor;
import com.bjpowernode.mvc.interceptor.LoginInterceptor;
import org.springframework.context.annotation.Configuration;
import org.springframework.format.FormatterRegistry;
import org.springframework.web.servlet.config.annotation.InterceptorRegistry;
import org.springframework.web.servlet.config.annotation.ViewControllerRegistry;
import org.springframework.web.servlet.config.annotation.WebMvcConfigurer;
/**
* springmvc配置类
*/
@Configuration
public class MvcSettings implements WebMvcConfigurer {
//页面跳转控制器,从请求直达视图
@Override
public void addViewControllers(ViewControllerRegistry registry) {
//配置页面控制:addViewController("请求的uri").指定他的视图setViewName(目标视图)
registry.addViewController("/welcome").setViewName("index");
}
@Override
public void addFormatters(FormatterRegistry registry) {
registry.addFormatter(new DeviceFormatter());
}
@Override
public void addInterceptors(InterceptorRegistry registry) {
//权限拦截器
AuthInterceptor authInterceptor = new AuthInterceptor();
registry.addInterceptor(authInterceptor)
.order(2)
.addPathPatterns("/article/**")
.excludePathPatterns("/article/query");
LoginInterceptor loginInterceptor = new LoginInterceptor();
}
}
step4:测试拦截器
POST http://localhost:8080/article/add
Content-Type: application/x-www-form-urlencoded
loginUser=lisi&title=Vue3
(二)、多个拦截器
增加一个验证登录用户的拦截器,只有 zhangsan,lisi,admin 能够登录系统。其他用户不可以。
两个拦截器登录的拦截器先执行,权限拦截器后执行,order()方法设置顺序,整数值越小,先执行。 step1:创建登录拦截器
step1:创建登录拦截器
import jakarta.servlet.http.HttpServletRequest;
import jakarta.servlet.http.HttpServletResponse;
import org.springframework.util.StringUtils;
import org.springframework.web.servlet.HandlerInterceptor;
import java.util.ArrayList;
import java.util.Arrays;
import java.util.List;
public class LoginInterceptor implements HandlerInterceptor {
private List<String> permitUser = new ArrayList<>();
public LoginInterceptor() {
this.permitUser = Arrays.asList("zhangsan", "lisi", "admin");
}
@Override
public boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler) throws Exception {
System.out.println("=========LoginInterceptor执行了==========");
//获取登录用户名
String loginUser = request.getParameter("loginUser");
if(StringUtils.hasText(loginUser) && permitUser.contains(loginUser)){
return true;
}
return false;
}
}
step2:登记拦截器,设置顺序 order
@Override
public void addInterceptors(InterceptorRegistry registry) {
//权限拦截器
AuthInterceptor authInterceptor = new AuthInterceptor();
registry.addInterceptor(authInterceptor)
.order(2)
.addPathPatterns("/article/**")
.excludePathPatterns("/article/query");
LoginInterceptor loginInterceptor = new LoginInterceptor();
registry.addInterceptor(loginInterceptor)
.order(1)
.addPathPatterns("/**") // 拦截所有请求
.excludePathPatterns("/article/query")//排除/article/query 请求;
}
step3:测试拦截器
POST http://localhost:8080/article/add
Content-Type: application/x-www-form-urlencoded
loginUser=lisi&title=Vue3