Spring Boot 如何优雅的校验参数
前言
做web开发有一点很烦人就是要校验参数,基本上每个接口都要对参数进行校验,比如一些格式校验 非空校验都是必不可少的。如果参数比较少的话还是容易 处理的一但参数比较多了的话代码中就会出现大量的 IF ELSE就比如下面这样:

这个例子只是校验了一下空参数。如果需要验证邮箱格式和手机号格式校验的话代码会更多,所以介绍一下 validator通过注解的方式进行校验参数。
1 | <!--版本自行控制,这里只是简单举例--> |
注解介绍
validator内置注解
| 注解 | 详细信息 |
|---|---|
@Null | 被注释的元素必须为 null |
@NotNull | 被注释的元素必须不为 null |
@AssertTrue | 被注释的元素必须为 true |
@AssertFalse | 被注释的元素必须为 false |
@Min(value) | 被注释的元素必须是一个数字,其值必须大于等于指定的最小值 |
@Max(value) | 被注释的元素必须是一个数字,其值必须小于等于指定的最大值 |
@DecimalMin(value) | 被注释的元素必须是一个数字,其值必须大于等于指定的最小值 |
@DecimalMax(value) | 被注释的元素必须是一个数字,其值必须小于等于指定的最大值 |
@Size(max, min) | 被注释的元素的大小必须在指定的范围内 |
@Digits (integer, fraction) | 被注释的元素必须是一个数字,其值必须在可接受的范围内 |
@Past | 被注释的元素必须是一个过去的日期 |
@Future | 被注释的元素必须是一个将来的日期 |
@Pattern(value) | 被注释的元素必须符合指定的正则表达式 |
Hibernate Validator 附加的 constraint
| 注解 | 详细信息 |
|---|---|
@Email | 被注释的元素必须是电子邮箱地址 |
@Length | 被注释的字符串的大小必须在指定的范围内 |
@NotEmpty | 被注释的字符串的必须非空 |
@Range | 被注释的元素必须在合适的范围内 |
@NotBlank | 验证字符串非null,且长度必须大于0 |
注意:
@NotNull适用于任何类型被注解的元素必须不能与NULL@NotEmpty适用于String Map或者数组不能为Null且长度必须大于0@NotBlank只能用于String上面 不能为null,调用trim()后,长度必须大于0
使用
模拟用户注册封装了一个 UserDTO
当提交数据的时候如果使用以前的做法就是 IF ELSE判断参数使用 validator则是需要增加注解即可。
例如非空校验:

然后需要在 controller方法体添加 @Validated不加 @Validated校验会不起作用

然后请求一下请求接口,把 Email参数设置为空
参数:
1 | { |
返回结果:

后台抛出异常

这样是能校验成功,但是有个问题就是返回参数并不理想,前端也并不容易处理返回参数,所以我们添加一下全局异常处理,然后添加一下全局统一返回参数这样比较规范。
添加全局异常
创建一个 GlobalExceptionHandler类,在类上方添加 @RestControllerAdvice注解然后添加以下代码:
1 | /** |
此方法主要捕捉 MethodArgumentNotValidException异常然后对异常结果进行封装,如果需要在自行添加其他异常处理。
添加完之后我们在看一下运行结果,调用接口返回:
1 | { |
OK 已经对异常进行处理。
校验格式
如果想要校验邮箱格式或者手机号的话也非常简单。
校验邮箱
1 | /** |
使用正则校验手机号
校验手机号使用正则进行校验,然后限制了一下位数
1 | /** |
查看一下运行结果
传入参数:
1 | { |
返回结果:
1 | { |
自定义注解
上面的注解只有这么多,如果有特殊校验的参数我们可以使用 Validator自定义注解进行校验
首先创建一个 IdCard注解类
1 |
|
在 UserDTO中添加 @IdCard注解即可验证,在运行时触发,本文不对自定义注解做过多的解释,下篇文章介绍自定义注解
- message 提示信息
- groups 分组
- payload 针对于Bean
然后添加 IdCardValidator 主要进行验证逻辑

上面调用了 is18ByteIdCardComplex方法,传入参数就是手机号,验证身份证规则自行百度
然后使用
1 |
|
分组
就比如上面我们定义的 UserDTO中的参数如果要服用的话怎么办?
在重新定义一个类然后里面的参数要重新添加注解?
Validator提供了分组方法完美了解决 DTO服用问题
现在我们注册的接口修改一下规则,只有用户名不能为空其他参数都不进行校验
先创建分组的接口
1 | public interface Create extends Default { |
我们只需要在注解加入分组参数即可例如:
1 | /** |
然后在修改Controller在@Validated中传入Create.class
1 | @PostMapping("/user") |
然后调用传入参数:
1 | { |
返回参数:
1 | { |
OK 现在只对Create的进行校验,而 Updata组的不校验,如果需要复用 DTO的话可以使用分组校验
校验单个参数
在开发的时候一定遇到过单个参数的情况,在参数前面加上注解即可
1 |
|
然后在 Controller类上面增加 @Validated注解,注意不是增加在参数前面。
作者:孙罗蒙









