本篇是关于springboot的参数校验知识,当然也适用其它java应用;读完本篇将学会基本的参数校验,自定义参数校验和分组参数校验;良好的代码规范和书写方式犹如散文版清丽脱俗,行云流水;
公众号:知识追寻者
知识追寻者(Inheriting the spirit of open source, Spreading technology knowledge;)
springboot在 web启动器中已经包含validator包
<dependency>
<groupId>org.springframework.bootgroupId>
<artifactId>spring-boot-starter-webartifactId>
dependency>
非springboot项目,需要自行引入依赖
<dependency>
<groupId>org.hibernate.validatorgroupId>
<artifactId>hibernate-validatorartifactId>
<version>6.1.5.Finalversion>
dependency>
<dependency>
<groupId>org.glassfishgroupId>
<artifactId>jakarta.elartifactId>
<version>3.0.3version>
dependency>
@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(regex=,flag=)
:指定的正则表达式@URL
:必须是一个URL@Email
:必须是email格式@NotBlank
: 字符串不能为空@NotEmpty
:集合不能为空@Length
: 长度必须在指定范围内@Valid
:对实体类进行校验@Valid
注解;/**
* @Author lsc
*
*/
@Data
public class SysUser {
private Long id;
@NotNull(message ="用户名不能为空")
@Size(min = 3, max = 5, message = "用户名长度为{min}-{max}之间")
private String username;
@NotNull(message ="昵称不能为空")
private String name;
@NotNull(message ="密码不能为空")
private String password;
@Email(message = "邮箱格式不合法")
private String email;
private String gender;
}
@Validated
;@Valid
注解/**
* @Author lsc
*
*/
@RestController
@Validated
public class SysUserController {
// 方法参数为实体校验
@PostMapping("/register")
public ResponseEntity register(@Valid @RequestBody SysUser sysUser){
return ResponseEntity.ok(sysUser);
}
// 方法参数校验
@GetMapping("user")
public ResponseEntity getUser(@NotNull(message ="用户名不能为空") String username) {
SysUser sysUser = new SysUser();
sysUser.setName("知识追寻者");
return ResponseEntity.ok(sysUser);
}
}
全局异常捕获,当出现参数校验不合法时捕获异常,并且返回给前端;
/**
* @Author lsc
*
*/
@ControllerAdvice
public class GlobHandler {
// 捕获方法参数校验异常
@ExceptionHandler(ConstraintViolationException.class)
@ResponseBody
public ResponseEntity constraintViolationExceptionHandler(ConstraintViolationException e){
Set<ConstraintViolation<?>> message = e.getConstraintViolations();
HashMap<String, Object> map = new HashMap<>();
message.stream().forEach(msg -> {
String path = msg.getPropertyPath().toString();
String field = path.substring(path.indexOf(".")+1);
map.put(field,msg.getMessageTemplate());
});
return ResponseEntity.ok(map);
}
// 捕获实体参数校验异常
@ExceptionHandler(MethodArgumentNotValidException.class)
@ResponseBody
public ResponseEntity resolveMethodArgumentNotValidException(MethodArgumentNotValidException e){
List<ObjectError> allErrors = e.getBindingResult().getAllErrors();
HashMap<String, Object> map = new HashMap<>();
allErrors.stream().forEach(error -> {
FieldError fieldError = (FieldError) error;
map.put(fieldError.getField(), fieldError.getDefaultMessage());
});
return ResponseEntity.ok(map);
}
}
特殊的字段需要自定义规则,比如身份证号码,邮箱,电话等;
定义校验注解
@Target({ METHOD, FIELD })
@Retention(RUNTIME)
@Constraint(validatedBy = GenderValidator.class)
@Documented
public @interface Gender {
String message() default "性别为男或者女";
Class<?>[] groups() default { };
Class<? extends Payload>[] payload() default {};
}
GenderValidator 实现 ConstraintValidator 接口并提供校验规则
/**
* @Author lsc
*
*/
public class GenderValidator implements ConstraintValidator<Gender, String> {
// 初始化校验值
@Override
public void initialize(Gender constraintAnnotation) {
}
// 校验规则
@Override
public boolean isValid(String value, ConstraintValidatorContext context) {
return "男".equals(value) || "女".equals(value);
}
}
在成员变量使用 注解
@Gender()
private String gender;
默认情况下,不指定分组都属于默认组;使用分组校验有利于分层校验开发;
新建2个接口,一个用于查询, 一个用于添加
/**
* @Author lsc
*
*/
public interface ADD extends Default {
}
public interface Select extends Default {
}
修改 实体校验规则,如果不指定分组 默认是 Default 组;
@NotNull(message ="用户名不能为空",groups = ADD.class)
@Size(min = 3, max = 5, message = "用户名长度为{min}-{max}之间")
private String username;
@NotNull(message ="昵称不能为空", groups = Select.class)
private String name;
@NotNull(message ="密码不能为空",groups = ADD.class)
private String password;
@Email(message = "邮箱格式不合法", groups = Select.class)
private String email;
控制层示例,此时只会校验 ADD 组和 Default组
// 分组校验
@PostMapping("/user")
public ResponseEntity addUser(@Validated(value = ADD.class) @RequestBody SysUser sysUser){
return ResponseEntity.ok(sysUser);
}
https://docs.jboss.org/hibernate/stable/validator/reference/en-US/html_single/
https://zhuanlan.zhihu.com/p/96403211
https://blog.csdn.net/justry_deng/article/details/86571671
https://juejin.im/post/5dc8bc745188254e7a155ba0
https://www.jianshu.com/p/89a675b7c900
Spring Boot 面试精讲