SpringMVC校验注解不生效

简介

SpringMVC框架提供了SPI扩展:javax.validation.spi.ValidationProvider,用来实现参数校验功能。Spring使用hibernate-validator作为它的默认实现,我们只需要进行一些简单的注解声明,就可以达到参数校验的功能。但是在实际使用场景中,经常会出现校验没生效的问题。

原因分析

  1. 检查jar包依赖。
    需要确报项目引入hibernate-validator以及与之匹配的validation-api版本,推荐直接依赖spring-boot-starter-validation
xml 复制代码
<dependency>
	<groupId>org.springframework.boot</groupId>
	<artifactId>spring-boot-starter-validation</artifactId>
</dependency>

最好检查下pom实际生效的依赖,有可能因为项目中显示引入了这两个包或者其他包间接依赖了这两个包导致他们版本不匹配。hibernate-validator依赖的validator-api才是它支持的版本。

  1. 检查注解是否正确

    新版本J2EE已经把javax包改成了jarkata包,检查项目中的validation-api中校验注解的包路径,选择与之匹配的注解。
    javax.validation.constraints.NotNull, jakarta.validation.constraints.NotNull

  2. 检查方法签名中,参数签名是否加了@Valid注解,必须加了@Valid注解的参数才会被校验。

  3. 检查参数是不是List类型,目前List不会被校验。

  4. 检查复杂字段上是否有@Valid注解,成员变量如果不是基本类型,需要在上面使用@Valid注解。

  5. 其他情况可以打断点排查:org.hibernate.validator.internal.engine.ValidatorImpl#validate

解决方案

对于参数是List类型,需要自己新建一个ValidationUtil类,在方法开始处手动调用校验。

java 复制代码
public class ValidationUtil {

    private static final Validator validator;

    static {
        validator = Validation.buildDefaultValidatorFactory().getValidator();
    }

    public static void validate(Object object, String objectName) {
        if (object instanceof List) {
            for (Object obj : (List<?>) object) {
                validateEntity(obj, objectName);
            }
        } else {
            validateEntity(object, objectName);
        }
    }

	@SneakyThrows
    private static void validateEntity(Object object, String objectName) {
        Set<ConstraintViolation<Object>> constraintViolations = validator.validate(object);
        if (!constraintViolations.isEmpty()) {
            throw new MethodArgumentNotValidException(null, createBindingResult(constraintViolations, objectName));
        }
    }

    private static BindingResult createBindingResult(Set<ConstraintViolation<Object>> violations, String objectName) {
        List<FieldError> fieldErrors = new ArrayList<>();

        for (ConstraintViolation<Object> violation : violations) {
            FieldError fieldError = new FieldError(
                    objectName,
                    violation.getPropertyPath().toString(),
                    violation.getMessage()
            );
            fieldErrors.add(fieldError);
        }

        return new BeanPropertyBindingResult(null, objectName, false, 256) {
            @NotNull
            @Override
            public List<FieldError> getFieldErrors() {
                return fieldErrors;
            }
        };
    }
}

// 调用方法
ValidationUtil.validate(paramValue, "param name");
相关推荐
油丶酸萝卜别吃3 分钟前
什么是 Java 内存模型(JMM)?
java·开发语言
量子炒饭大师19 分钟前
【C++入门】Cyber神经的义体插件 —— 【类与对象】内部类
java·开发语言·c++·内部类·嵌套类
Hx_Ma1623 分钟前
测试题(四)
java·开发语言·jvm
Never_Satisfied36 分钟前
在c#中,抛出异常,并指定其message的值
java·javascript·c#
没有bug.的程序员40 分钟前
IDEA 效能巅峰实战:自定义模板 Live Templates 内核、快捷键精密逻辑与研发提效深度指南
java·ide·intellij-idea·快捷键·研发提效·自定义模板
追随者永远是胜利者1 小时前
(LeetCode-Hot100)22. 括号生成
java·算法·leetcode·职场和发展·go
逝水如流年轻往返染尘1 小时前
java中的泛型
java
百锦再1 小时前
Java重入锁(ReentrantLock)全面解析:从入门到源码深度剖析
java·开发语言·struts·spring·kafka·tomcat·intellij-idea
知识即是力量ol1 小时前
口语八股—— Spring 面试实战指南(终篇):常用注解篇、Spring中的设计模式
java·spring·设计模式·面试·八股·常用注解
yuezhilangniao1 小时前
win10环境变量完全指南:Java、Maven、Android、Flutter -含我的环境备份
android·java·maven