案例引入
下面我们以新增一个员工为功能切入点,以常规写法为背景,慢慢烘托出 @Valid 和 @Validated 注解用法详解。
那么,首先,我们会有一个员工对象 Employee,如下 :
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31
|
public class Employee { public String name; public Integer age; public String getName() { return name; } public void setName(String name) { this.name = name; } public Integer getAge() { return age; } public void setAge(Integer age) { this.age = age; } }
|
然后 Cotroller 中会有一个对应都新增方法 add()
,如下:
1 2 3 4 5 6 7 8 9 10 11
| @Controller public class TestController { @RequestMapping("/add") @ResponseBody public String add(Employee employee) { return "新增员工成功"; } }
|
现在要求:员工的名称不能为空,且长度不能超过10个字符,那么我们以前的做法大致如下:
写完,我们启动项目测试下:
(1)名称为空情况
(2)正常情况
(3)超过长度情况
可以看到,和我们料想中的一样,毫无问题。
除了名称外,我们规定年龄也是必填项,且范围在1到100岁,那么此时,我们需要增加对应判定代码如下:
那么问题来了,现在员工对象 Employee 就 2 个字段,我们就写了 10 多行的代码验证,要是有20个字段,岂不是要写 100 多行代码?通常来说,当一个方法中的无效业务代码量过多时,往往代码设计有问题,当然这不是我们所想看到都结果。
那么如何解决呢?首先大家应该会想到将对应的验证过程抽成一个验证方法,如下:
这样来看,我们的业务方法就清爽多了。
但这种方式只是抽了一个方法,有一种换汤不换药的感觉,虽然业务方法看起来清爽了很多,但书写代码量并没有下降,反而还多出了一个方法,这也不是我们理想中的样子。
@Valid 详解
此时,我们引出 Spring 中的 @valid 注解,这些问题就可以迎刃而解了,具体如下:
首先,我们在 Maven 配置中引入 @valid 的依赖:
如果你是 springboot 项目,那么可以不用引入了,已经引入了,他就存在于最核心的 web 开发包里面。
1 2 3 4 5
| <dependency> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-starter-web</artifactId> <version>2.0.5.RELEASE</version> </dependency>
|
如果你不是 springboot 项目,那么引入下面依赖即可:
1 2 3 4 5 6 7 8 9 10 11
| <dependency> <groupId>javax.validation</groupId> <artifactId>validation-api</artifactId> <version>1.1.0.Final</version> </dependency> <dependency> <groupId>org.hibernate</groupId> <artifactId>hibernate-validator</artifactId> <version>5.4.1.Final</version> </dependency>
|
那么针对上面情景,我们可以对我们的代码进行优化了。
首先我们在 Employee 类的属性上打上如下注解:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43
| package com.zyq.beans; import javax.validation.constraints.NotBlank; import javax.validation.constraints.NotNull; import org.hibernate.validator.constraints.Length; import org.hibernate.validator.constraints.Range;
public class Employee { @NotBlank(message = "请输入名称") @Length(message = "名称不能超过个 {max} 字符", max = 10) public String name; @NotNull(message = "请输入年龄") @Range(message = "年龄范围为 {min} 到 {max} 之间", min = 1, max = 100) public Integer age; public String getName() { return name; } public void setName(String name) { this.name = name; } public Integer getAge() { return age; } public void setAge(Integer age) { this.age = age; } }
|
然后再 Controller 对应方法上,对这个员工标上 @Valid 注解,表示我们对这个对象属性需要进行验证,
既然验证,那么就肯定会有验证结果,所以我们需要用一个东西来存放验证结果,做法也很简单,在参数直接添加一个BindingResult,具体如下:
对应获取验证结果的代码如下:
需要注意的是@Valid和BindingResult是一一对应的,如果有多个@Valid,那么每个@Valid后面都需要添加BindingResult用于接收bean中的校验信息
1 2 3 4 5 6
| FieldError fieldError = bindingResult.getFieldError();
String field = fieldError.getField();
String defaultMessage=fieldError.getDefaultMessage();
|
OK ! 万事俱备 !我们进行测试下:
(1)名称为空
(2)名称正常,年龄为空
(3)名称超出范围,年龄正常
(4)名称正常,年龄超出范围
可以看到,代码不但简洁了很多,结果和预期的也一模一样!很棒吧!!
常用注解:
除了刚刚都注解,最后再附加2个常用注解,我就直接贴图了,基本上这6个注解可以解决99%的字段,其他注解我就不贴图了,如果不满足,自己问百度。
@Validated 详解
上面,我们讲述了 @Valid 注解,现在我们来说说 @Validated 这个注解,在我看来,@Validated 是在 @Valid 基础上,做的一个升级版。
我们可以看到,我们在使用 @Valid 进行验证的时候,我们需要用一个对象去接收校验结果,最后根据校验结果判断,从而提示用户。
如果我们把手动校验的这段代码删除或注释掉,那么即使当我们的字段不满足规则时,方法种的程序也是能够被执行的。
比如,我们将字段值置空时,正常情况是会进行提示的。
当我们把校验逻辑注释掉后,再次执行上面的请求后。
可以看到我们的程序继续往后面去执行完成了。
现在,我们去掉方法参数上的 @Valid 注解和其配对的 BindingResult 对象,
然后再校验的对象前面添加上 @Validated 注解。
这个时候,我们再次请求,可以看到,我们请求报400错误了。
而我们通过程序的异常日志来看,提示说是 age 和 name 字段为了空,致使请求失败。
那么,从这里我们可以得知,当我们的数据存在校验不通过的时候,程序就会抛出
org.springframework.validation.BindException 的异常。
在实际开发的过程中,我们肯定不能讲异常直接展示给用户,而是给能看懂的提示。
于是,我们不妨可以通过捕获异常的方式,将该异常进行捕获。
首先我们创建一个校验异常捕获类 ValidExceptionHandler ,然后打上 @RestControllerAdvice 注解,该注解表示他会去抓所有 @Controller 标记类的异常,并在异常处理后返回以 JSON 或字符串的格式响应前端。
在异常捕捉到后,我们同上面的 @valid 校验一样,只返回第一个错误提示。
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15
| package com.zyq.config; import org.springframework.validation.BindException; import org.springframework.web.bind.annotation.ExceptionHandler; import org.springframework.web.bind.annotation.RestControllerAdvice; @RestControllerAdvice public class ValidExceptionHandler { @ExceptionHandler(BindException.class) public String validExceptionHandler(BindException exception) { return exception.getAllErrors().get(0).getDefaultMessage(); } }
|
那么,我们现在重启程序,然后重新请求,就可以发现界面已经不报400错误了,而是直接提示了我们的错误信息。
@Valid 和 @Validated 比较
最后我们来对 @Valid 和 @Validated 两个注解进行总结下:
(1)@Valid 和 @Validated 两者都可以对数据进行校验,待校验字段上打的规则注解(@NotNull, @NotEmpty等)都可以对 @Valid 和 @Validated 生效;
(2)@Valid 进行校验的时候,需要用 BindingResult 来做一个校验结果接收。当校验不通过的时候,如果手动不 return ,则并不会阻止程序的执行;
(3)@Validated 进行校验的时候,当校验不通过的时候,程序会抛出400异常,阻止方法中的代码执行,这时需要再写一个全局校验异常捕获处理类,然后返回校验提示。
(4)总体来说,@Validated 使用起来要比 @Valid 方便一些,它可以帮我们节省一定的代码,并且使得方法看上去更加的简洁。
实际测试发现:使用@Valid进行校验的时候,不使用 BindingResult 接收校验结果,在统一异常处理时,也能拦截到MethodArgumentNotValidException错误。
自定义校验注解:
创建注解:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15
| @Target({ElementType.METHOD,ElementType.FIELD}) @Retention(RetentionPolicy.RUNTIME) @Constraint(validatedBy = {MyConstraintValidator.class}) public @interface MyValid {
int min() default 3;
String message() default "{javax.validation.constraints.NotBlank.message}";
Class<?>[] groups() default { };
Class<? extends Payload>[] payload() default { }; }
|
创建自定义验证规则文件,简单的实现:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28
|
public class MyConstraintValidator implements ConstraintValidator<MyValid, Object> {
@Override public void initialize(MyValid constraintAnnotation) { int min = constraintAnnotation.min(); System.out.println("Exception"); }
@Override public boolean isValid(Object o, ConstraintValidatorContext constraintValidatorContext) { if (o.toString().length()>min){ return true; } return false; } }
|
实体类:
1 2 3 4
| public class Person { @MyValid(min = 3,message = "姓名长度不能低于3") private String name; }
|
@Validated分组功能
比@Valid多了分组功能,用法和@Valid相同,并且自定义注解也一样
扩展性高,可以根据不同的业务需求,根据设置分组进行校验
1 2 3 4 5 6 7 8 9 10 11
| public interface aa { } public interface bb { }
public class Person { @NotBlank(message = "name不能为空",groups = aa.class) private String name; @NotBlank(message = "id不能为空",groups = bb.class) private String id; }
|
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17
| @RequestMapping("/valid") public void valid(@RequestBody @Validated(aa.class) Person person,BindingResult bindingResult) { Map<String, Object> model = bindingResult.getModel(); } 只会校验name
@RequestMapping("/valid") public void valid(@RequestBody @Validated(bb.class) Person person,BindingResult bindingResult) { Map<String, Object> model = bindingResult.getModel(); } 只会校验id
@RequestMapping("/valid") public void valid(@RequestBody @Validated(aa.class,bb.class) Person person,BindingResult bindingResult) { Map<String, Object> model = bindingResult.getModel(); } 会校验id和name
|
注意:分组的class只能为接口
嵌套类校验
1 2 3
| @Validated:可以用在类型、方法和方法参数上。但是不能用在成员属性(字段)上 @Valid:可以用在方法、构造函数、方法参数和成员属性(字段)上 两者是否能用于成员属性(字段)上直接影响能否提供嵌套验证的功能
|
嵌套验证:
1 2 3 4 5 6 7 8
| public class Person { @NotBlank(message = "name不能为空") private String name; @NotBlank(message = "id不能为空") private String id; @NotNull(message = "stus不能为空") private List<Stu> stus; }
|
嵌套类就是一个类中包含另一个类
1 2 3 4 5 6
| public class Stu { @NotBlank(message = "stu_name不能为空") private String stu_name; @NotBlank(message = "stu_id不能为空") private String stu_id; }
|
如上图Person类中只能校验stus,不能校验Stu类中的stu_name和stu_id
如果想要嵌套验证,必须在Person的stus字段上面标明这个字段的实体也要进行验证,由于@Validated不能用在成员属性(字段)上,但是@Valid能加在成员属性(字段)上,而且@Valid类注解上也说明了它支持嵌套验证功能,那么我们能够推断出:@Valid加在方法参数时并不能够自动进行嵌套验证,而是用在需要嵌套验证类的相应字段上,来配合方法参数上@Validated或@Valid来进行嵌套验证。
例如:
1 2 3 4 5 6 7 8 9 10 11
| public class Person { @NotBlank(message = "name不能为空") private String name; @NotBlank(message = "id不能为空") private String id;
@Valid @NotNull(message = "stus不能为空") private List<Stu> stus; }
|
就可以校验实体类Person中字段stus对应实体类中的属性了。
1 2 3 4
| @RequestMapping("/valid") public void valid(@RequestBody @Valid Person person,BindingResult bindingResult) { Map<String, Object> model = bindingResult.getModel(); }
|
@Validated: 用在方法入参上无法单独提供嵌套验证功能。不能用在成员属性(字段)上,也无法提示框架进行嵌套验证。能配合嵌套验证注解@Valid进行嵌套验证。
@Valid: 用在方法入参上无法单独提供嵌套验证功能。能够用在成员属性(字段)上,提示验证框架进行嵌套验证。能配合嵌套验证注解@Valid进行嵌套验证。
requestParam/PathVariable参数校验
GET请求一般会使用requestParam/PathVariable传参。如果参数比较多(比如超过6个),还是推荐使用DTO对象接收。
否则,推荐将一个个参数平铺到方法入参中。在这种情况下,必须在Controller类上标注@Validated注解(添加 @Valid 不会生效),并在入参上声明约束注解(如@Min等)。如果校验失败,会抛出ConstraintViolationException
异常。
代码示例如下:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26
| @RequestMapping("/api/user") @RestController @Validated public class UserController { @GetMapping("{userId}") public Result detail(@PathVariable("userId") @Min(10000000000000000L) Long userId) { UserDTO userDTO = new UserDTO(); userDTO.setUserId(userId); userDTO.setAccount("11111111111111111"); userDTO.setUserName("xixi"); return Result.ok(userDTO); } @GetMapping("getByAccount") public Result getByAccount(@Length(min = 6, max = 20) @NotNull String account) { UserDTO userDTO = new UserDTO(); userDTO.setUserId(10000000000000003L); userDTO.setAccount(account); userDTO.setUserName("hello"); return Result.ok(userDTO); } }
|
全局统一异常处理
在实际项目开发中,@Validated通常会用统一异常处理来返回一个更友好的提示。比如我们系统要求无论发送什么异常,http的状态码必须返回对应业务码,由业务码去区分系统的异常情况。
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29
| import java.util.List; import org.springframework.validation.BindException; import org.springframework.validation.BindingResult; 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; @RestControllerAdvice public class CommonExceptionHandler { @ExceptionHandler({MethodArgumentNotValidException.class}) public Result handleMethodArgumentNotValidException(MethodArgumentNotValidException ex) { BindingResult bindingResult = ex.getBindingResult(); StringBuilder sb = new StringBuilder("校验失败:"); for (FieldError fieldError : bindingResult.getFieldErrors()) { sb.append(fieldError.getField()).append(":").append(fieldError.getDefaultMessage()).append(", "); } String msg = sb.toString(); return Result.fail(BusinessCode.参数校验失败, msg); } @ExceptionHandler({ConstraintViolationException.class}) public Result handleConstraintViolationException(ConstraintViolationException ex) { return Result.fail(BusinessCode.参数校验失败, ex.getMessage()); } }
|
上文说 @Valid进行校验的时候,需要用 BindingResult 来做一个校验结果接收,其实可以使用AOP技术来做全局统一异常处理。方法如下:
第一步,在需要验证的字段上加上 Hibernate Validator 提供的校验注解。
比如说我现在有一个用户名和密码登录的请求参数 UsersLoginParam 类:
1 2 3 4 5 6 7 8 9 10 11 12 13
| @Data @ApiModel(value="用户登录", description="用户表") public class UsersLoginParam implements Serializable { private static final long serialVersionUID = 1L;
@ApiModelProperty(value = "登录名") @NotBlank(message="登录名不能为空") private String userLogin;
@ApiModelProperty(value = "密码") @NotBlank(message="密码不能为空") private String userPass; }
|
第二步,在对应的请求接口(UsersController.login()
)中添加 @Validated
注解,并注入一个 BindingResult
参数。
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21
| @Controller @Api(tags="用户") @RequestMapping("/users") public class UsersController { @Autowired private IUsersService usersService;
@ApiOperation(value = "登录以后返回token") @RequestMapping(value = "/login", method = RequestMethod.POST) @ResponseBody public ResultObject login(@Validated UsersLoginParam users, BindingResult result) { String token = usersService.login(users.getUserLogin(), users.getUserPass()); if (token == null) { return ResultObject.validateFailed("用户名或密码错误"); } Map<String, String> tokenMap = new HashMap<>(); tokenMap.put("token", token); tokenMap.put("tokenHead", tokenHead); return ResultObject.success(tokenMap); } }
|
第三步,为控制层(UsersController)创建一个切面,将通知注入到 BindingResult 对象中,然后再判断是否有校验错误,有错误的话返回校验提示信息,否则放行。
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27
| @Aspect @Component @Order(2) public class BindingResultAspect { @Pointcut("execution(public * com.codingmore.controller.*.*(..))") public void BindingResult() { }
@Around("BindingResult()") public Object doAround(ProceedingJoinPoint joinPoint) throws Throwable { Object[] args = joinPoint.getArgs(); for (Object arg : args) { if (arg instanceof BindingResult) { BindingResult result = (BindingResult) arg; if (result.hasErrors()) { FieldError fieldError = result.getFieldError(); if(fieldError!=null){ return ResultObject.validateFailed(fieldError.getDefaultMessage()); }else{ return ResultObject.validateFailed(); } } } } return joinPoint.proceed(); } }
|
可以看得出,Hibernate Validator 带来的优势有这些:
- 验证逻辑与业务逻辑进行了分离,降低了程序耦合度;
- 统一且规范的验证方式,无需再次编写重复的验证代码。
不过,也带来一些弊端,比如说:
- 需要在请求接口的方法中注入 BindingResult 对象,而这个对应在方法体中并没有用到
- 只能校验一些非常简单的逻辑,涉及到数据查询就无能为力了。
处理比较复杂的逻辑校验,可以在校验失败的时候直接抛出自定义异常,然后进行捕获处理就可以了。
实际开发中把两者结合在一起用,就可以弥补彼此的短板了,简单校验用 Hibernate Validator,复杂一点的逻辑校验,比如说需要数据库查询用全局自定义异常来实现。
为了把 Spring Boot 逻辑校验这块单独拉出来做一个 demo 的例子,我新建了一个 codingmore-validator 的项目,放在 codingmore-learning 项目下面了,想要参考的,直接戳下面的两个链接。
完整的编程喵整个项目的源码戳第一个,只想看逻辑校验的戳第二个。
Hibernate-Validator注解
常用注解
注解 |
使用 |
@NotNull |
被注释的元素(任何元素)必须不为 null, 集合为空也是可以的。没啥实际意义 |
@NotEmpty |
用来校验字符串、集合、map、数组不能为null或空 (字符串传入空格也不可以)(集合需至少包含一个元素) |
@NotBlank |
只用来校验字符串不能为null,空格也是被允许的 。校验字符串推荐使用@NotEmpty |
@Size(max=, min=) |
指定的字符串、集合、map、数组长度必须在指定的max和min内 允许元素为null,字符串允许为空格 |
@Length(min=,max=) |
只用来校验字符串,长度必须在指定的max和min内 允许元素为null |
@Range(min=,max=) |
用来校验数字或字符串的大小必须在指定的min和max内 字符串会转成数字进行比较,如果不是数字校验不通过 允许元素为null |
@Min() |
校验数字(包括integer short long int 等)的最小值,不支持小数即double和float 允许元素为null |
@Max() |
校验数字(包括integer short long int 等)的最小值,不支持小数即double和float 允许元素为null |
@Pattern() |
正则表达式匹配,可用来校验年月日格式,是否包含特殊字符(regexp = “^[a-zA-Z0-9\u4e00-\u9fa5 |
除了@Empty要求字符串不能全是空格,其他的字符串校验都是允许空格的。
message是可以引用常量的,但是如@Size里max不允许引用对象常量,基本类型常量是可以的。
注意大部分规则校验都是允许参数为null,即当不存在这个值时,就不进行校验了
不常用注解
注解 |
说明 |
@Null |
被注释的元素必须为 null |
@AssertTrue |
被注释的元素必须为 true |
@AssertFalse |
被注释的元素必须为 false |
@DecimalMin(value) |
被注释的元素必须是一个数字,其值必须大于等于指定的最小值 |
@DecimalMax(value) |
被注释的元素必须是一个数字,其值必须小于等于指定的最大值 |
@Digits (integer, fraction) |
被注释的元素必须是一个数字,其值必须在可接受的范围内 |
@Past |
被注释的元素必须是一个过去的日期 |
@Future |
被注释的元素必须是一个将来的日期 |
@Email |
被注释的元素必须是电子邮箱地址 |