评论

收藏

[Java] 快速校验实体类时,@Valid,@Validated,@NotNull注解无效的解决

编程语言 编程语言 发布于:2022-02-28 17:00 | 阅读数:1042 | 评论:0

这篇文章主要介绍了快速校验实体类时,@Valid,@Validated,@NotNull注解无效的解决方案,具有很好的参考价值,希望对大家有所帮助。如有错误或未考虑完全的地方,望不吝赐教。
校验实体类参数内容不能为空时使用注解校验无效

使用@valid注解首先引入依赖
如果是SpringBoot项目,引入web开发包,就不需要再单独引入@valid依赖了、因为他存在于Web开发包中的最核心之中
<dependency>
  <groupId>org.springframework.boot</groupId>
  <artifactId>spring-boot-starter-web</artifactId>
  <version>2.0.5.RELEASE</version>
</dependency>
如果不是SpringBoot项目,要在Maven的Pom中显式引入@valid依赖
<dependency>
  <groupId>javax.validation</groupId>
  <artifactId>validation-api</artifactId>
  <version>1.1.0.Final</version>
</dependency>
 
<dependency>
  <groupId>org.hibernate</groupId>
  <artifactId>hibernate-validator</artifactId>
  <version>5.4.1.Final</version>
</dependency>
校验实体类大部分人使用的方法肯定都是controller层里面拿出来判断是否为空,其实可以直接通过实体类中的注释@NotNull直接校验,如图
DSC0000.png

但经过我的实验,不知道什么原因,在我的项目中并没有起到作用,解决如下

1、更换方法入参类型
在网上看到@Valid是可以用在方法、构造函数、方法参数和成员属性(字段)上,而我一开始controller层中使用的是json接收的,转换成实体类后再校验的,如图
DSC0001.png

修改如下:在controller接收数据时直接以对象接收,在入口中直接以注解进行校验
DSC0002.png


2、错误内容过多
此时对象字段校验生效,但是出错时返回的是很凌乱的内容,如图:
DSC0003.png


3、使用对象接收错误内容,按自己要求输出
上面的图片可看出,返回内容中defaultMessage就是想要输出的内容,所以在方法中添加一个接收错误信息的类型为BindingResult的对象,如果对象有数据,则输出错误数据,如图:
DSC0004.png

另一层面的实现了实体类的参数校验,大家如果有更好的方法可以留言交流

springboot 校验机制 @Validated @Valid

1、探究原因
在开发的过程中一直迷惑 @Validated 与 @Valid 的用法,有时候是@Validated ,有时候是@Valid 。虽然能够实现校验,但是还是不够明确何时能够生效,不了解他生效的情况
首先定位2个注解所属的包:
@Validated 在 spring-context 包下属于spring 提供的核心包
DSC0005.png

@Valid 在 validation-api 包下 2.0.2 版本
DSC0006.png

@Validated 是spring 核心包,是每个项目都有的,那么 api 是如何引入的? 查看maven 依赖
DSC0007.png

原来是在引入 Spring-boot-start-web 的时候,就引入了该依赖
两个注解存在不同的包,而@NotNull ,@Null ,@Size ,@Max 等校验注解是哪里的呢?
DSC0008.png

这些注解都是在 api 包下

2、使用@Validated 实现校验机制
情景一: 查询参数是一个实体,Get 请求,在不添加任何注解的情况下,查询是正常的,实体参数字段都为null
现在需求 id 字段不能为空,在实体id 字段标记 @NotNull ,继续查询,发现注解没有生效
经过测试,只有请求实体参数列表前加@Validated 才会生效,即使@Validated 加在类上也无法生效
情景二: 查询参数是基本或者引用类型字段,参数列表中加入 @NutNull 修改该字段。发现无法生效
经过测试:只有全局类上加@Validated 才会生效,即使参数列表中加入 @Validated 也无法生效
产生异常也有所不同:在校验生效的情况下,实体类校验产生的异常是:BindException , 而参数列表产生的异常是:ConstraintViolationException

3、使用@Valid 实现校验机制
场景一:与上述一致,只有@Valid 作用在参数列表前才会生效
场景二:@Valida 不管是左右在参数列表还是类上,都无法生效。只能使用@Validated 全局设置
结论:暂不清楚@Valid 设计出现的原因,所以的校验@Validate 均可以实现
附加全局异常捕获:
@RestControllerAdvice
public class GlobalException {
  @ExceptionHandler({BindException.class})
  public RespResult validationException(BindException exception){
    List<ObjectError> errors =  exception.getAllErrors();
    if(!CollectionUtils.isEmpty(errors)){
      StringBuilder sb = new StringBuilder();
      errors.forEach(e->sb.append(e.getDefaultMessage()).append(","));
      return new RespResult(400, sb.toString());
    }
    return new RespResult(500, exception.getLocalizedMessage());
  }
  @ExceptionHandler({ConstraintViolationException.class})
  public RespResult constraintViolationException(ConstraintViolationException exception){
    Set<ConstraintViolation<?>> constraintViolations = exception.getConstraintViolations();
    if(!CollectionUtils.isEmpty(constraintViolations)){
      StringBuilder sb = new StringBuilder();
      constraintViolations.forEach(e->sb.append(e.getMessage()).append(","));
      return new RespResult(400, sb.toString());
    }
    return new RespResult(500, exception.getLocalizedMessage());
  }
}
以上为个人经验,希望能给大家一个参考,也希望大家多多支持CodeAE代码之家
原文链接:https://blog.csdn.net/qq_41444892/article/details/103427461

关注下面的标签,发现更多相似文章