一、 为什么要用分组校验?(痛点)
在日常开发中,我们经常遇到同一个 DTO(数据传输对象)在不同场景下有不同校验规则的情况。
典型场景:用户管理
- 新增用户:
ID必须为空(因为是自增的),但Password必填。 - 修改用户:
ID不能为空(指定改谁),但Password选填(不改就不传)。
如果没有分组校验,我们通常有两种“笨办法”:
- 定义两个 DTO:
UserCreateReqVO和UserUpdateReqVO。- 缺点:类爆炸,大量重复字段,维护麻烦。
- 去掉校验注解:在 DTO 里不写
@NotNull,全靠在 Service 层手写if (id == null)判断。- 缺点:代码臃肿,失去了注解校验的优雅。
分组校验(Groups)就是为了解决这个问题——用一个 DTO,搞定多种场景。
二、 核心步骤(用法总结)
使用分组校验只需要三步:定义分组 -> 标记规则 -> 触发校验。
1. 定义分组接口 (The Marker)
这只是一个普通的 Java 接口,不需要任何方法,它的唯一作用就是做一个“标签”。
publicclassUserReqVO{// 定义两个分组接口publicinterfaceCreate{}publicinterfaceUpdate{}}2. 在字段上标记分组 (The Rule)
在校验注解中配置groups属性。
@DatapublicclassUserReqVO{@Schema(description="用户ID")@Null(groups=Create.class,message="新增时ID必须为空")@NotNull(groups=Update.class,message="修改时ID不能为空")privateLongid;@Schema(description="用户名")@NotBlank(message="用户名不能为空")// 没有指定 groups,属于默认分组 (Default)privateStringusername;@Schema(description="密码")@NotBlank(groups=Create.class,message="新增时密码不能为空")privateStringpassword;}3. 触发校验 (The Trigger)
这是最关键的一步。你必须告诉 Spring,当前这次请求,你要检查哪个分组。
注意:必须使用@Validated注解,@Valid不支持分组。
@RestController@RequestMapping("/user")publicclassUserController{// 场景 A:新增 -> 触发 Create 分组@PostMapping("/create")publicResultcreate(@RequestBody@Validated(UserReqVO.Create.class)UserReqVOreq){returnsuccess(userService.create(req));}// 场景 B:修改 -> 触发 Update 分组@PutMapping("/update")publicResultupdate(@RequestBody@Validated(UserReqVO.Update.class)UserReqVOreq){returnsuccess(userService.update(req));}}三、 进阶:如何处理“默认规则”?(坑点预警)
这里有一个初学者极容易踩的坑。
问题:
在上面的例子中,username字段只加了@NotBlank,没有加groups。
当你使用@Validated(Create.class)时,Spring 只会检查标记了Create.class的字段,username会被忽略!
解决方案:让分组接口继承Default
如果你希望在检查Create分组时,也顺便检查那些没有分组的默认字段,可以这样定义接口:
importjavax.validation.groups.Default;publicclassUserReqVO{// 让 Create 分组继承 Default 分组publicinterfaceCreateextendsDefault{}publicinterfaceUpdateextendsDefault{}}这样,@Validated(Create.class)就会同时检查:
- 标记了
Create的规则。 - 没有标记 group 的规则(归属于
Default)。
四、 高阶实战:业务逻辑中的动态校验
除了在 Controller 自动校验,我们有时需要在 Service 层根据业务开关动态校验(例如:验证码开关)。
场景:系统配置了“开启验证码”开关。开启时校验,关闭时不校验。
代码示例:
@ServicepublicclassAuthService{@ResourceprivateValidatorvalidator;// 注入 Java 标准校验器publicvoidlogin(LoginReqVOreq){// 1. 动态判断业务开关if(isCaptchaEnabled()){// 2. 手动触发特定分组 (CodeEnableGroup) 的校验Set<ConstraintViolation<LoginReqVO>>errors=validator.validate(req,CodeEnableGroup.class);// 3. 如果有异常,手动抛出if(!errors.isEmpty()){thrownewServiceException("验证码不能为空");}}// ... 其他逻辑}}五、 总结
Validation Groups (分组校验)是 Spring Boot 开发中精简代码的神器。
- 本质:给校验注解加上“条件判断”。
- 核心:通过空接口定义分组 (
interface GroupA {})。 - 使用:在注解中指定
groups = GroupA.class。 - 触发:
- 自动:Controller 层使用
@Validated(GroupA.class)。 - 手动:Service 层使用
validator.validate(obj, GroupA.class)。
- 自动:Controller 层使用
- 最佳实践:让自定义分组继承
Default接口,避免漏掉通用校验规则。 - 缺点:创建与更新复用一个类,高度耦合;需要在Controller中指定class参数;无法实现复杂的方法级别的逻辑校验。有一种平替方案可以见@AssertTrue。