在 Spring Boot 中,我们可以使用 Hibernate Validator(Bean Validation 的参考实现)来实现参数校验。
它的核心思路是:把校验逻辑从业务代码里抽离出来,用注解的方式声明校验规则。
接下来我们一步步来看怎么实现。
首先,定义一个用于接收用户注册参数的 DTO 对象:
@Data
public class UserRegistrationRequest {
@NotNull(message = "用户名不能为空")
@Size(min = 3, max = 20, message = "用户名长度必须在3到20之间")
private String username;
@NotNull(message = "密码不能为空")
@Size(min = 8, message = "密码长度至少为8个字符")
private String password;
@NotNull(message = "年龄不能为空")
@Min(value = 1, message = "年龄必须是正整数")
@Max(value = 120, message = "年龄不能超过120")
private Integer age;
@Email(message = "邮箱格式不正确")
private String email;
}
这里我们用了几个常见的校验注解:
@NotNull
:字段不能为空;@Size
:限制字符串长度;@Min
和 @Max
:限制数值范围;@Email
:校验邮箱格式。这些注解由 Hibernate Validator 提供,基本涵盖了日常开发中的大部分校验需求。
然后,在 Controller 中这样写:
@RestController
@RequestMapping("/api/users")
public class UserController {
@PostMapping("/register")
public ResponseEntity<String> register(@Valid @RequestBody UserRegistrationRequest request) {
return ResponseEntity.ok("注册成功!");
}
}
注意这里的 @Valid
注解,它的作用是告诉 Spring:对请求参数进行校验。
如果前端传的参数不合法,Spring 会抛出一个 MethodArgumentNotValidException
异常。默认情况下,这个异常返回的信息不太友好,可能是这样的:
{
"timestamp": "2024-01-01T12:00:00.000+00:00",
"status": 400,
"error": "Bad Request",
"message": "Validation failed for object='userRegistrationRequest'. Error count: 2",
"path": "/api/users/register"
}
为了提升用户体验,我们可以用全局异常处理器来统一格式化错误信息:
@RestControllerAdvice
public class GlobalExceptionHandler {
@ExceptionHandler(MethodArgumentNotValidException.class)
public ResponseEntity<Map<String, String>> handleValidationException(MethodArgumentNotValidException ex) {
Map<String, String> errors = new HashMap<>();
ex.getBindingResult().getFieldErrors().forEach(error -> {
errors.put(error.getField(), error.getDefaultMessage());
});
return ResponseEntity.badRequest().body(errors);
}
}
现在,当参数校验失败时,返回的错误信息会变成这样:
{
"username": "用户名长度必须在3到20之间",
"password": "密码不能为空"
}
清晰又直观,用户一看就明白自己错在哪儿了。
有些场景下,不同的接口对参数的校验规则是不一样的,比如:
username
和 password
是必填项;email
和 age
。这种情况下,可以用 分组校验 来解决。
public interface RegisterGroup {}
public interface UpdateGroup {}
public class UserRequest {
@NotNull(groups = RegisterGroup.class, message = "用户名不能为空")
@Size(min = 3, max = 20, groups = RegisterGroup.class, message = "用户名长度必须在3到20之间")
private String username;
@NotNull(groups = RegisterGroup.class, message = "密码不能为空")
private String password;
@Email(groups = UpdateGroup.class, message = "邮箱格式不正确")
private String email;
@Min(value = 1, groups = UpdateGroup.class, message = "年龄必须是正整数")
private Integer age;
}
@RestController
@RequestMapping("/api/users")
public class UserController {
@PostMapping("/register")
public ResponseEntity<String> register(@Validated(RegisterGroup.class) @RequestBody UserRequest request) {
return ResponseEntity.ok("注册成功!");
}
@PutMapping("/update")
public ResponseEntity<String> update(@Validated(UpdateGroup.class) @RequestBody UserRequest request) {
return ResponseEntity.ok("更新成功!");
}
}
如果 Hibernate Validator 提供的注解不能满足需求,还可以自定义校验注解。例如,校验手机号格式。
@Documented
@Constraint(validatedBy = PhoneValidator.class)
@Target({ElementType.FIELD})
@Retention(RetentionPolicy.RUNTIME)
public @interface ValidPhone {
String message() default "手机号格式不正确";
Class<?>[] groups() default {};
Class<? extends Payload>[] payload() default {};
}
public class PhoneValidator implements ConstraintValidator<ValidPhone, String> {
private static final String PHONE_REGEX = "^1[3-9]\\d{9}$";
@Override
public boolean isValid(String value, ConstraintValidatorContext context) {
return value != null && value.matches(PHONE_REGEX);
}
}
public class UserRequest {
@ValidPhone
private String phone;
}
优雅的参数校验不仅能提高代码的可维护性,还能显著提升用户体验。
在 Spring Boot 中,通过使用 Hibernate Validator 提供的注解,配合分组校验、自定义校验和统一异常处理。
我们可以轻松实现简洁、高效、可扩展的参数校验机制。
优雅的参数校验的秘籍是: