简单使用
- 要在Springboot项目中加入参数校验功能首先得加入
spring-boot-starter-validation
依赖
<dependency><groupId>org.springframework.boot</groupId><artifactId>spring-boot-starter-validation</artifactId></dependency>
//
- 然后给需要校验的字段添加上约束性注解,如我们对实体类参数进行校验
publicclassValidEntity{ privateintid; privateStringappId; privateStringname; privateStringemail; }
常见约束注解如下:
注:此表格只是简单的对注解功能的说明,并没有对每一个注解的属性进行说明;可详见源码。
- 在Controller层对需要参数校验的方法加上@Validated注解
参数校验一般分为两类:在Controller使用模型接收数据时, @Validated注解直接放在该模型参数前即可。
value="test1") (publicStringtest1(ValidEntityvalidEntity){ return"test1 valid success"; } value="test3") (publicStringtest3(ValidEntityvalidEntity){ return"test3 valid success"; }
当我们是直接在Controller层中的参数前,使用约束注解时,@Validated要直接放在类上
value="test2") (publicStringtest2(Stringemail){ return"test2 valid success"; }
此时需要在主类上增加@Validated注解
"/demo/valid") (publicclassValidController { ... }
在参数校验时我们既可以使用@Validated也可以使用@Valid注解,两者功能大部分类似;
主要区别在于:
@Valid属于javax下的,而@Validated属于spring下;
@Valid支持嵌套校验、而@Validated不支持,@Validated支持分组,而@Valid不支持。
统一异常处理
如果参数校验未通过Spring会抛出三种类型的异常
- 当对@RequestBody需要的参数进行校验时会出现
org.springframework.web.bind.MethodArgumentNotValidException
- 当直接校验具体参数时会出现
javax.validation.ConstraintViolationException
,也属于ValidationException
异常
- 当直接校验对象时会出现
org.springframework.validation.BindException
在SpringBoot中统一拦截处理只需要在配置类上添加@RestControllerAdvice
注解,然后在具体方法中通过@ExceptionHandler
指定需要处理的异常,具体
代码如下:
publicclassGlobalExceptionHandler { publicstaticfinalStringERROR_MSG="系统异常,请联系管理员。"; value= {BindException.class, ValidationException.class, MethodArgumentNotValidException.class}) (publicResponseEntity<Result<String>>handleValidatedException(Exceptione) { Result<String>resp=null; if (einstanceofMethodArgumentNotValidException) { // BeanValidation exceptionMethodArgumentNotValidExceptionex= (MethodArgumentNotValidException) e; resp=newResult<>(Integer.toString(HttpStatus.BAD_REQUEST.value()), ex.getBindingResult().getAllErrors().stream().map(ObjectError::getDefaultMessage).collect(Collectors.joining(", ")) , getStackTrace(ex)); } elseif (einstanceofConstraintViolationException) { // BeanValidation GET simple paramConstraintViolationExceptionex= (ConstraintViolationException) e; resp=newResult<>(Integer.toString(HttpStatus.BAD_REQUEST.value()), ex.getConstraintViolations().stream().map(ConstraintViolation::getMessage).collect(Collectors.joining(", ")) , getStackTrace(ex)); } elseif (einstanceofBindException) { // BeanValidation GET object paramBindExceptionex= (BindException) e; resp=newResult<>(Integer.toString(HttpStatus.BAD_REQUEST.value()), ex.getAllErrors().stream().map(ObjectError::getDefaultMessage).collect(Collectors.joining(", ")) , getStackTrace(ex)); } returnnewResponseEntity<>(resp,HttpStatus.BAD_REQUEST); } privateStringgetStackTrace(Exceptione) { //打印日志开关,可通过配置读取booleanprintStrackTrace=false; if(printStrackTrace){ StringWritersw=newStringWriter(); e.printStackTrace(newPrintWriter(sw)); returnsw.toString(); }else{ returnERROR_MSG; } } }
最终实现效果如下:
参数分组
有下面一个实体类,我们需要对其进行参数校验。
publicclassValidEntity { privateintid; privateStringappId; privateStringname; privateStringemail; }
但是实际业务是在编辑的时候appId才是必填,在新增的时候name必填,这时候可以用groups分组功能来实现:同一个模型在不同场景下,动态区分校验模型中的不同字段。
使用方式
首先我们定义一个分组接口ValidGroup,再在分组接口总定义出多个不同的操作类型,Create,Update,Query,Delete
publicinterfaceValidGroupextendsDefault{ interfaceCrudextendsValidGroup{ interfaceCreateextendsCrud{ } interfaceUpdateextendsCrud{ } interfaceQueryextendsCrud{ } interfaceDeleteextendsCrud{ } } }
这里的ValidGroup
继承了Default,当然也可以不继承,具体区别我们后面再说。
- 在模型中给校验参数分配分组
value="ValidEntity") (publicclassValidEntity { privateintid; groups=ValidGroup.Crud.Update.class) (privateStringappId; groups=ValidGroup.Crud.Create.class) (privateStringname; privateStringemail; }
tips:这里@Email注解未指定分组,默认会属于Default分组,appId和name指定了分组就不会再属于Default分组了。
- 在参数校验时通过value属性指定分组
这里通过@Validated(value = ValidGroup.Crud.Update.class)指定了具体的分组,上面提到的是否继承Default的区别在于:
如果继承了Default,@Validated标注的注解也会校验未指定分组或者Default分组的参数,比如email
如果不继承Default则不会校验未指定分组的参数,需要加上@Validated(value = {ValidGroup.Crud.Update.class, Default.class}才会校验
快速失败(Fali Fast)
默认情况下在对参数进行校验时Spring Validation会校验完所有字段然后才抛出异常,可以通过配置开启Fali Fast模式,一旦校验失败就立即返回。
publicclassValidatedConfig { publicValidatorvalidator() { ValidatorFactoryvalidatorFactory=Validation.byProvider(HibernateValidator.class) .configure() // 快速失败模式 .failFast(true) .buildValidatorFactory(); returnvalidatorFactory.getValidator(); } }