一、基础概述
1.简介
Java API 规范 (JSR303) 定义了 Bean 校验的标准 validation-api
,但没有提供实现。hibernate validation
是对这个规范的实现,并增加了校验注解如 @Email、@Length 等。Spring Validation
是对 hibernate validation
的二次封装,用于支持 spring mvc 参数自动校验。
2.依赖
<dependency><groupId>org.springframework.boot</groupId><artifactId>spring-boot-starter-validation</artifactId>
</dependency>
如果spring-boot
版本小于 2.3.x,spring-boot-starter-web 会自动传入 hibernate-validator
依赖。如果 spring-boot 版本大于 2.3.x,则需要手动引入依赖
二、参数效验
对于 web 服务来说,为防止非法参数对业务造成影响,在 Controller 层一定要做参数校验的!大部分情况下,请求参数分为如下两种形式:
POST
、PUT
请求,使用@RequestBody
传递参数
GET
请求,使用 @RequestParam
/ @PathVariable
传递参数
1.@RequestParam参数校验
@RestController
// 代表需要参数验证,一定要加
@Validated
public class HelloController {// @Min代表参数不能小于10@RequestMapping(value = "/test1")public Object m1(@RequestParam(value = "number") @Min(value = 10) Integer number) {System.out.println(number);return UUID.randomUUID() + "----" + number;}}
【测试】
http://localhost:8081/test1?number=9 参数为9即报错,异常为ConstraintViolationException
http://localhost:8081/test1?number=12 参数为12即正常
2.@PathVariable参数效验
@RestController
@Validated
public class HelloController {@RequestMapping(value = "/test2/{number}")public Object m2(@PathVariable(value = "number") @Max(value = 20) String number) {System.out.println(number);return UUID.randomUUID() + "----" + number;}}
3.@RequestBody参数效验
在实体类上生命效验字段
package com.h3c.entity;import lombok.Data;
import org.hibernate.validator.constraints.Length;
import javax.validation.constraints.NotNull;@Data
public class UserParam {@NotNullprivate String userName;@Length(min = 6, max = 20, message = "长度范围为6~20")private String account;@Length(min = 6, max = 20)private String password;}
在方法参数上声明校验注解@Validated
@RestController
public class HelloController {// 需要设置声明,@Valid和@Validated都可以@PostMapping(value = "/test3")public Object m3(@RequestBody @Validated UserParam param) {System.out.println(param);return UUID.randomUUID() + "----" + param;}
}
参数错误会报异常MethodArgumentNotValidException
4.全局异常处理
前面说过,如果校验失败,会抛出MethodArgumentNotValidException
或者ConstraintViolationException
异常。在实际项目开发中,通常会用统一异常处理来返回一个更友好的提示。
package com.h3c.exception;import com.h3c.entity.Result;
import org.springframework.validation.FieldError;
import org.springframework.web.bind.MethodArgumentNotValidException;
import org.springframework.web.bind.annotation.ExceptionHandler;
import org.springframework.web.bind.annotation.RestControllerAdvice;
import javax.validation.ConstraintViolationException;
import java.util.LinkedHashMap;
import java.util.List;
import java.util.Map;@RestControllerAdvice
public class SystemExceptionHandler {// MethodArgumentNotValidException异常可以获取到异常字段@ExceptionHandler({MethodArgumentNotValidException.class})public Result handleMethodArgumentNotValidException(MethodArgumentNotValidException ex) {ex.printStackTrace();// 获取所有的错误字段List<FieldError> errors = ex.getBindingResult().getFieldErrors();Map<String, String> map = new LinkedHashMap<>();// 把错误信息存入一个map,然后返回errors.forEach(item -> {map.put(item.getField(), item.getDefaultMessage());});return Result.error(map);}@ExceptionHandler({ConstraintViolationException.class})public Result handleConstraintViolationException(ConstraintViolationException ex) {ex.printStackTrace();return Result.error(ex.getMessage());}
}
【POST请求参数异常返回示例】
【GET请求参数异常返回示例】
{"flag":false,"code":500,"message":"m1.number: 最小不能小于10","data":null}
三、效验注解
https://www.cnblogs.com/jinzlblog/p/16635043.html 注解
注解 | 用法 | 适用类型 |
---|---|---|
@Null | 被注解的字段必须为空 | |
@NotNull | 被注解的字段必须不为空 | |
@NotBlank | 带注解的元素不能为null,并且必须至少包含一个非空白字符 | |
@NotEmpty | 带注解的元素不能为null也不能为空 | String(长度)集合(大小)数组(长度) |
@AssertTrue | 检查该字段必须为True | Boolean |
@AssertFalse | 检查该字段必须为False | Boolean |
@Min(value) | 被注解的字段必须大于等于指定的最小值 | |
@Max(value) | 被注解的字段必须小于等于指定的最大值 | |
@Negative | 带注解的元素必须是严格的负数(0被认为是无效值) | BigDecimal,BigInteger,byte,short,int,long及其包装类 |
@NegativeOrZero | 带注解的元素必须是严格的负数或0 | BigDecimal,BigInteger,byte,short,int,long及其包装类 |
@Positive | 带注解的元素必须是严格的正数(0被认为是无效值) | BigDecimal,BigInteger,byte,short,int,long及其包装类 |
@PositiveOrZero | 带注解的元素必须是严格的正数或0 | BigDecimal,BigInteger,byte,short,int,long及其包装类 |
@DecimalMin | 被注解的字段必须大于等于指定的最小值 | BigDecimal,BigInteger,byte,short,int,long及其包装类 |
@DecimalMax | 被注解的字段必须小于等于指定的最大值 | BigDecimal,BigInteger,byte,short,int,long及其包装类 |
@Size(min=,max=) | 被注解的字段的size必须在min和max之间,不需要判空 | 字符串、数组、集合 |
@Digits(integer, fraction) | 被注解的字段必须在指定范围内,整数部分长度小于integer,小数部分长度小于fraction | 字符串、数组、集合 |
@Past | 被注解的字段必须是一个过去的日期时间 | |
@PastOrPresent | 被注解的字段必须是过去的或现在的日期时间 | |
@Future | 被注解的字段必须是一个将来的日期时间 | |
@FutureOrPresent | 被注解的字段必须是现在的或将来的日期时间 | |
字符串必须是格式正确的电子邮件地址 | String | |
@Pattern(value) | 被注解的字段必须符合指定的正则表达式 |
四、高级使用
1.分组校验
在实际项目中,可能多个接口需要使用同一个 DTO 类来接收参数,而不同方法的校验规则很可能是不一样的。这个时候,简单地在 DTO 类的字段上加约束注解无法解决这个问题。因此,spring-validation
支持了分组校验的功能,专门用来解决这类问题。
还是上面的例子,比如保存 User 的时候,UserId 是可空的,但是更新 User 的时候,UserId 的值必须存在,其它字段的校验规则在两种情况下一样。这个时候就需要使用分组校验
简单点说:就是根据设置的条件来执行参数校验,其实就是一个判断,筛选设置了分组的参数进行校验,不过在这里叫做分组
【声明分组】
// 保存的时候校验分组
public interface InsertValidGroup {
}// 更新的时候校验分组
public interface UpdateValidGroup {
}
【实体类】
需要注意的是,这里面只有参数Id设置了分组,其它参数没有设置则不会进行校验
@Data
public class UserParam {// 代表在属于更新的时候,校验id@NotNull(groups = UpdateValidGroup.class)private Integer id;@NotNullprivate String userName;@Length(min = 6, max = 20, message = "长度范围为6~20")private String account;@Length(min = 6, max = 20)private String password;}
【接口】
@RestController
@Validated
public class HelloController {@PostMapping(value = "/test3")public Object m3(@RequestBody @Validated(UpdateValidGroup.class) UserParam param) {System.out.println(param);return UUID.randomUUID() + "----" + param;}}
【测试】
请求体id为空,则出现异常,请求体id存在,则正常执行
【其余参数设置分组】
如上所示只有设置了分组的参数才会校验,那么其它的参数怎么办呢,只能挨个写
@Data
public class UserParam {@NotNull(groups = UpdateValidGroup.class)private Integer id;// 其余的参数把全部的分组都设置上@NotNull(groups = {UpdateValidGroup.class, InsertValidGroup.class})private String userName;@NotNull(groups = {UpdateValidGroup.class, InsertValidGroup.class})@Length(min = 6, max = 20, message = "长度范围为6~20")private String account;@NotNull(groups = {UpdateValidGroup.class, InsertValidGroup.class})@Length(min = 6, max = 20)private String password;
}
2.嵌套校验
当我们实体类中某个字段是对象,这种情况下,可以使用嵌套校验
@Data
public class UserParam {@NotNull(groups = UpdateValidGroup.class)private Integer id;@NotNull(groups = {UpdateValidGroup.class, InsertValidGroup.class})private String userName;@NotNull(groups = {UpdateValidGroup.class, InsertValidGroup.class})@Length(min = 6, max = 20, message = "长度范围为6~20")private String account;@NotNull(groups = {UpdateValidGroup.class, InsertValidGroup.class})@Length(min = 6, max = 20)private String password;// 可以针对对象参数校验@NotNull(groups = {UpdateValidGroup.class, InsertValidGroup.class})@Validprivate Job job;// 可以针对集合对象参数校验@NotNull(groups = {UpdateValidGroup.class, InsertValidGroup.class})@Validprivate List<Job> jobs;@Datapublic static class Job {@NotNull(groups = UpdateValidGroup.class)@Min(value = 10, groups = UpdateValidGroup.class)private Long jobId;@NotNull(groups = {UpdateValidGroup.class, InsertValidGroup.class})private String jobName;@NotNull(groups = {UpdateValidGroup.class, InsertValidGroup.class})private String position;}}
【接口情况】
3.集合校验
【参数类】
@Data
public class ValidList {// 集合校验@Valid@NotNull@Size(min = 3, max = 10)private List<String> list;}
【接口】
@RestController
@Validated
public class HelloController {@PostMapping(value = "/test4")public Object m4(@RequestBody @Validated ValidList list) {System.out.println(list);return UUID.randomUUID() + "----" + list;}}
【测试】
4.自定义校验
业务需求总是比框架提供的这些简单校验要复杂的多,我们可以自定义校验来满足我们的需求。
例如性别参数只能是0或者1,某个字段必须是{“aa”,“bb”,"cc}中的一个
【自定义注解】
@Target({METHOD, FIELD, ANNOTATION_TYPE, CONSTRUCTOR, PARAMETER, TYPE_USE})
@Retention(RUNTIME)
@Documented
// 这里需要注意标明校验类
@Constraint(validatedBy = {ProcessValidator.class})
public @interface CheckValid {String message() default "数据错误";Class<?>[] groups() default {};Class<? extends Payload>[] payload() default {};}
【校验实现类】
需要实现ConstraintValidator
接口,然后重写isValid()
,验证该参数必须属于集合内
public class ProcessValidator implements ConstraintValidator<CheckValid, String> {private List<String> list = Arrays.asList("aa", "bb", "cc");@Overridepublic boolean isValid(String value, ConstraintValidatorContext context) {if (value != null) {if (list.contains(value)) {return true;}}return false;}}
【实体类参数】
@Data
public class ForumParam {// 通过正则表达式验证@NotNull@Pattern(regexp = "^(男|女){1}$")private String sex;// 通过自定义注解校验@NotNull@CheckValidprivate String pms;}
【接口】
@RestController
@Validated
public class HelloController {@PostMapping(value = "/test4")public Object m4(@RequestBody @Validated ForumParam param) {System.out.println(param);return UUID.randomUUID() + "----" + param;}}
【测试】
5.编程式校验
上面的示例都是基于注解来实现自动校验的,在某些情况下,我们可能希望以编程方式调用验证。这个时候可以注入 javax.validation.Validator 对象,然后再调用其 api。
@Autowired
private javax.validation.Validator globalValidator;// 编程式校验
@PostMapping(value = "/test5")
public Object m5(@RequestBody UserParam param) {Set<ConstraintViolation<UserParam>> set = validator.validate(param, UpdateValidGroup.class);// 如果校验通过,set;否则,set包含未校验通过项if (set.isEmpty()) {// 校验通过,才会执行业务逻辑处理} else {// 遍历出现异常的字段for (ConstraintViolation<UserParam> violation : set) {System.out.println(violation.getPropertyPath() + "---" + violation.getMessage());}}System.out.println(param);return UUID.randomUUID() + "----" + param;
}
6.快速失败
Spring Validation 默认会校验完所有字段,然后才抛出异常。可以通过一些简单的配置,开启 Fali Fast 模式,一旦校验失败就立即返回。
package com.h3c.config;import org.hibernate.validator.HibernateValidator;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;import javax.validation.Validation;
import javax.validation.Validator;
import javax.validation.ValidatorFactory;/*** @version JDK11* @author: wys4822* @date: 2022年09月07日*/
@Configuration
public class ValidConfig {@Beanpublic Validator validator() {ValidatorFactory validatorFactory = Validation.byProvider(HibernateValidator.class).configure()// 快速失败模式.failFast(true).buildValidatorFactory();return validatorFactory.getValidator();}}
【测试】
7.@Valid 和 @Validated 区别
五、源码分析
1.@RequestBody参数校验实现原理
a>RequestResponseBodyMethodProcessor
在 Spring-MVC
框架中,RequestResponseBodyMethodProcessor
是用于解析 @RequestBody
标注的参数以及处理 @ResponseBody
标注方法的返回值的。显然,执行参数校验的逻辑肯定就在解析参数的方法 resolveArgument()
中:
public class RequestResponseBodyMethodProcessor extends AbstractMessageConverterMethodProcessor {@Overridepublic Object resolveArgument(MethodParameter parameter, @Nullable ModelAndViewContainer mavContainer,NativeWebRequest webRequest, @Nullable WebDataBinderFactory binderFactory) throws Exception {parameter = parameter.nestedIfOptional();// 根据请求体转换参数Object arg = readWithMessageConverters(webRequest, parameter, parameter.getNestedGenericParameterType());String name = Conventions.getVariableNameForParameter(parameter);if (binderFactory != null) {WebDataBinder binder = binderFactory.createBinder(webRequest, arg, name);if (arg != null) {// 执行参数校验validateIfApplicable(binder, parameter);if (binder.getBindingResult().hasErrors() && isBindExceptionRequired(binder, parameter)) {throw new MethodArgumentNotValidException(parameter, binder.getBindingResult());}}if (mavContainer != null) {mavContainer.addAttribute(BindingResult.MODEL_KEY_PREFIX + name, binder.getBindingResult());}}return adaptArgumentIfNecessary(arg, parameter);}
}
b>validateIfApplicable()
protected void validateIfApplicable(WebDataBinder binder, MethodParameter parameter) {// 获取参数前面设置的注解,注解中肯定会包含@RequestBodyAnnotation[] annotations = parameter.getParameterAnnotations();for (Annotation ann : annotations) {// 判断是否存在@Validated,通过一个工具类来实现Object[] validationHints = ValidationAnnotationUtils.determineValidationHints(ann);if (validationHints != null) {binder.validate(validationHints);break;}}
}
c>determineValidationHints()
需要声明的是,在接口参数前面增加校验注解,注解可以为@Validated
或者@Valid
,2个都可以
@Valid
注解判断存在即可通过判断,而@Validated
注解判断存在后,还需要尝试获取里面的分组校验
public abstract class ValidationAnnotationUtils {private static final Object[] EMPTY_OBJECT_ARRAY = new Object[0];@Nullablepublic static Object[] determineValidationHints(Annotation ann) {Class<? extends Annotation> annotationType = ann.annotationType();String annotationName = annotationType.getName();// 通过注解路径判断是否为@Valid,原因是因为该注解有很多重名的if ("javax.validation.Valid".equals(annotationName)) {return EMPTY_OBJECT_ARRAY;}// 判断@Validated注解Validated validatedAnn = AnnotationUtils.getAnnotation(ann, Validated.class);// 因为@Validated可以设置分组校验,所以这里需要获取value,封装成数组返回if (validatedAnn != null) {Object hints = validatedAnn.value();return convertValidationHints(hints);}if (annotationType.getSimpleName().startsWith("Valid")) {Object hints = AnnotationUtils.getValue(ann);return convertValidationHints(hints);}return null;}private static Object[] convertValidationHints(@Nullable Object hints) {if (hints == null) {return EMPTY_OBJECT_ARRAY;}return (hints instanceof Object[] ? (Object[]) hints : new Object[]{hints});}}
d>validate()
上面验证了参数是否标注了@Validated
或者@Valid
,代表该参数需要执行校验逻辑,那么接下来肯定就是遍历字段校验了,那么该validate()
在上面的validateIfApplicable
被调用
public void validate(Object... validationHints) {// 获取到参数对象Object target = getTarget();// 参数为空,就报错了Assert.state(target != null, "No target to validate");// 获取默认的绑定结果BindingResult bindingResult = getBindingResult();// 遍历校验for (Validator validator : getValidators()) {if (!ObjectUtils.isEmpty(validationHints) && validator instanceof SmartValidator) {// 开启核心的校验((SmartValidator) validator).validate(target, bindingResult, validationHints);}else if (validator != null) {validator.validate(target, bindingResult);}}
}
2.@RequestParam参数校验实现原理
{// 获取到参数对象Object target = getTarget();// 参数为空,就报错了Assert.state(target != null, "No target to validate");// 获取默认的绑定结果BindingResult bindingResult = getBindingResult();// 遍历校验for (Validator validator : getValidators()) {if (!ObjectUtils.isEmpty(validationHints) && validator instanceof SmartValidator) {// 开启核心的校验((SmartValidator) validator).validate(target, bindingResult, validationHints);}else if (validator != null) {validator.validate(target, bindingResult);}}
}