@Valid和@Validated
@Valid和@Validated比较
相同点: @Valid 注解和 @Validated 注解都是开启校验功能的注解 不同点: @Validated 注解是 Spring 基于 @Valid 注解的进一步封装,并提供比如分组,分组顺序的高级功能 使用位置不同: @Valid 注解 : 可以使用在方法,构造函数,方法参数和成员属性上 @Validated 注解 : 可以用在类型,方法和方法参数上. 但是不能用在成员属性上
@Valid高级使用
@Valid级联校验 级联校验: 也叫嵌套检测.嵌套就是一个实体类包含另一个实体类 @Valid 和可以用在成员属性的字段上,因此 @Valid 可以提供级联校验 示例:
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 |
@Data public class Hair {
@NotBlank (message = "头发长度必须提交!" ) private Double length;
@NotBlank (message = "头发颜色必须提交!" ) private String color; }
@Data public class Person {
@NotBlank (message = "用户姓名必须提交!" ) @Size (min= 2 , max= 8 ) private String userName;
// 添加@Valid注解实现嵌套检测 @Valid @NotEmpty (message = "用户要有头发!" ) private List<Hair> hairs; }
@PostMapping ( "/person" ) public Result addPerson( @Valid @RequestBody Person person) { return Result.buildSuccess(person); } |
只是在方法参数前面添加 @Valid 和 @Validated 注解,不会对嵌套的实体类进行校验.要想实现对嵌套的实体类进行校验,需要在嵌套的实体类属性上添加 @Valid 注解
@Validated高级使用
@Validated分组校验 分组校验: 对指定的组开启校验,可以分别作用于不同的业务场景中 分组校验是由 @Validated 注解中的 value 提供的 groups: JSR 303 校验注解中的分组方法 groups 示例:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 |
@Data public class PersonGroup {
public interface AddGroup {}
public interface UpdateGroup {}
// @Validated注解value方法指定分组UpdateGroup.class时校验 @NotBlank (message = "用户ID必须提交!" , groups = UpdateGroup. class ) private String id;
// @Validated注解value方法指定分组AddGroup.class或者分组UpdateGroup.class时校验 @NotBlank (message = "用户的姓名必须提交!" , groups = {AddGroup. class , UpdateGroup. class }) private String name;
// @Validated注解value方法未指定分组时校验 @Range (min = 1 , max = 200 , message = "用户的年龄必须提交!" ) private int age; } |
开启分组校验: 通过 @Validated 注解的 value 方法对指定的分组开启校验
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 |
@RestController @RequestMapping ( "/person" ) public class PersonGroupController {
// 不指定分组时校验 @GetMapping ( "/person" ) public Result getPerson( @Validated @RequestBody PersonGroup person) { return Result.buildSuccess(person); }
// 指定AddGroup分组校验 @PostMapping ( "/person" ) public Result addPerson( @Validated (value = PersonGroup.AddGroup. class ) @RequestBody PersonGroup person) { return Result.buildSuccess(person); }
// 指定UpdateGroup分组校验 @PutMapping ( "/person" ) public Result updatePerson( @Validated (value = PersonGroup.updateGroup. class ) @RequestBody PersonGroup person) { return Result.buildSuccess(person); } } |
校验方法添加 groups 的值来指定分组,只有使用 @Validated 注解的 value 的值指定这个分组时,开会开启注解的校验数据的功能
@Validated分组校验顺序 默认情况下,分组间的约束是无序的,但是在一些特殊的情况下可能对分组间的校验有一定的顺序 比如第二组的分组的约束的校验需要依赖第一组的稳定状态来进行,此时,要求分组间的约束校验一定要有顺序 分组校验顺序通过使用 @GroupSequence 注解实现 示例:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 |
@Data public class UserGroupSequence {
public interface FirstGroup {}
public interface SecondGroup {}
// 使用GroupSequence定义分组校验顺序:按照FirstGroup,SecondGroup分组顺序进行校验 @GroupSequence ({FirstGroup. class , SecondGroup. class }) public interface Group {}
@NotEmpty (message = "用户ID必须提交!" , group = FirstGroup. class ) private String userId;
@NotEmpty (message = "用户姓名必须提交!" , group = FirstGroup. class ) @Size (min = 2 , max = 8 , message = "用户姓名的长度在2~8之间" , goup = Second. class ) private String userName; } |
1 2 3 4 5 6 7 8 9 |
@RestController @RequestMapping ( "/user" ) public class UserGroupSequenceController { // 这里方法中@Validated注解value的值是Group.class @PostMapping ( "/user" ) public Result addGroup( @Validated (value = Group. class ) @RequestBody UserGroupSequence user) { return Result.buildSuccess(user); } } |
使用 @GroupSequence 注解指定分组校验顺序后,第一组分组的约束的校验没有通过后,就不会进行第二组分组的约束的校验
@Validated非实体类校验 在非实体类上添加 @Validated 注解对非实体类进行校验
1 2 3 4 5 6 7 8 |
@Validated public class AnnotationController {
@GetMapping ( "/person" ) public Result getAge( @Range (min = 2 , max = 8 , message = "年龄在3~8岁!" ) @RequestParam int age) { return Result.buildSuccess(age); } } |
在 GlobalExceptionHandler 中添加全局统一异常处理方法:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 |
@ExceptionHandler (ConstraintViolationException. class ) @ResponseBody public Result resolveConstraintViolationException(ConstraintVilationException exception) { Set<ConstraintVilation<?>> constraintVilations = exception.getConstraintVilations(); // 处理异常信息 if (!CollectionUtils.isEmpty(constraintVilations)) { StringBuilder messageBuilder = new StringBuilder(); for (ConstraintVilation constraintViolation : constraintVilations) { messageBuilder.append(constraintVilation.getMessage()).append( "," ); } String errorMessage = messageBuilder.toString(); if (errorMessage.length() > 1 ) { errorMessage.substring( 0 , errorMessage.length() - 1 ); } return Result.builderFailure(ErrorStatus.ILLEGAL_DATA.getCode(), errorMessage); } return Result.builderFailure(ErrorStatus.ILLEGAL_DATA.getCode(), exception.getMessage()) } |
@PathVariable
@PathVariable的作用: 用来指定请求 URL 路径里面的变量 @PathVariable 和 @RequestParam 的区别: @PathVariable 用来指定请求 URL 中的变量 @RequestParam 用来获取静态的 URL 请求入参
正则表达式校验
使用正则表达式校验 @PathVariable 指定的路径变量
1 2 3 4 5 |
// 请求路径中的id必须是数字,否则寻找不到这个路径404 @GetMapping ( "/user/{id:\\d+}" ) public Result getId( @PathVariable (name= "id" ) String userId) { return Result.buildSuccess(userId); } |
继承BasicErrorController类
@ControllerAdvice 注解只能处理进入控制器方法抛出的异常 BasicErrorController 接口可以处理全局异常 @PathVariable 路径校验异常不是控制器方法抛出的,此时还没有进入控制器方法: BasicErrorController 处理异常,比如 404 异常时,会跳转到 /error 路径,此时会返回错误的 html 页面 为了保证返回结果统一,继承 BasicErrorController 类,重写 BasicErrorController 接口中的错误处理方法
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 |
@RestController public class PathErrorController extends BasicErrorController {
@Autowired public PathErrorController(ErrorAttributes errorAttributes, ServerProperties serverProperties, List<ErrorViewResolver> errorViewResolvers) { super (errorAttributes, serverProperties.getError(), errorViewResolvers); }
/** * 处理html请求 */ @Override public ModelAndView errorHtml(HttpServletRequest request, HttpServletResponse response) { HttpStatus status = getStatus(request); Map<String, Object> model = getErrorAttributes(request, isIncludeStackTrace(request, MediaType.TEXT_HTML)); ModelAndView modelAndView = new ModelAndView( "pathErrorPage" , model, status); return modelAndView; }
/** * 处理json请求 */ @Override public ResponseEntity<Map<String, Object>> error(HttpServletRequest request) { Map<String, Object> body = getErrorAttributes(request, isIncludeStackTrace(request, MediaType.ALL));
Map<String, Object> responseBody = new HashMap<>( 8 ); responseBody.put( "success" , false ); responseBody.put( "code" , body.get( "status" )); responseBody.put( "message" , body.get( "error" ));
return new ResponseEntity<>(responseBody, HttpStatus.OK); } } |
自定义校验注解
使用场景: 对某一个只能输入指定值的字段进行校验. 此时需要使用自定义注解实现 定义自定义的注解 @Show :
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 |
@Documented @Constraint (validateBy = {Show.ShowConstraintValidator. class }) @Target ({ElementType.METHOD, ElementType.FIELD, ElementType.ANNOTATION_TYPE}) @Rentation (RUNTIME) public @interface Show { String message() default "{com.oxford.annotation.Show.message}" ;
Class<?>[] groups() default {};
Class<? extends Payload>[] payload() default {};
int [] value();
class ShowConstraintValidator implements ConstraintValidator<Show, Integer> {
private Set<Integer> set = new HashSet<>();
/** * 初始化操作 * 获取value属性指定的数字,保存到Set集合中 */ @Override public void initilize(Show constraintAnnotation) { int [] value = constraintAnnotation.value(); for ( int v : value) { set.add(i); } }
@Override public boolean isValid(Integer value, ConstraintValidatorContext context) { return set.contains(value); } } } |
注意点: @Constraint 注解: 将自定义的注解和实现的校验类联系起来 自定义校验注解类需要实现 ConstraintValidator<A extends Annotation, T> 接口 接口中第一个泛型参数表示的是自定义注解类 接口中第二个泛型参数表示的是校验的属性的值的类型 initialize() 方法: 获取到自定义注解中的相关的数据 isValid() 方法: 实现自定义的校验逻辑 返回boolean类型的校验结果 自定义注解的使用:
1 2 3 4 5 6 |
@Data public class AnnotationQuery {
@Show (value = { 0 , 1 }, message = "数值只能是0或者1" ) private Integer isShow; } |
1 2 3 4 |
@PostMapping ( "/annotation" ) public Result addAnnotation( @Validated @RequestBody AnnotationQuery annotation) { return Result.buildSuccess(annotation); } |
到此这篇关于Java中的三种校验注解的使用(@Valid,@Validated和@PathVariable)的文章就介绍到这了,更多相关Java 校验注解内容请搜索以前的文章或继续浏览下面的相关文章希望大家以后多多支持!
原文链接:https://juejin.cn/post/7090327624970895397
查看更多关于Java中的三种校验注解的使用(@Valid,@Validated和@PathVariable)的详细内容...