首页
学习
活动
专区
圈层
工具
发布
社区首页 >问答首页 >Hibernate JSR303验证和错误生成的propertyPath

Hibernate JSR303验证和错误生成的propertyPath
EN

Stack Overflow用户
提问于 2011-04-20 10:44:39
回答 2查看 7.3K关注 0票数 4

我正在尝试使用Spring MVC设置表单的JSR-303验证。我已经正确地配置了所有内容(或者至少我认为我这样做了),并且验证大部分工作都是正确的。但是,如果我有一个命令对象,其中包含我想要验证的对象集合,并且我用@Valid注释了该集合,则Hibernate JSR-303提供程序没有提供正确的propertyPath。ContraintViolation对象中的propertyPath应该像list[0].barlist[1].bar一样填充,但Hibernate验证器只提供list[].barlist[].bar。当Spring的SpringValidatorAdaptor.validate()方法试图添加字段级错误时,这会导致NumberFormatException (因为它内部需要一个数字存在于这些括号中)。

使用spring-context-3.0.5和hibernate-validator-4.1.0.Final (我也尝试了4.0.2.GA和4.2.0.Beta2,得到了相同的结果),我写了一个小的单元测试来说明这个问题:

代码语言:javascript
复制
package com.foo;

import java.util.ArrayList;
import java.util.Collection;
import java.util.Set;

import javax.validation.ConstraintViolation;
import javax.validation.Valid;
import javax.validation.Validation;
import javax.validation.Validator;

import org.hibernate.validator.constraints.NotEmpty;
import org.junit.Test;
import org.springframework.util.AutoPopulatingList;

public class ValidatorTest {

    class Person {
        @NotEmpty
        private String name;

        @NotEmpty
        @Valid
        private Collection<Foo> list = new AutoPopulatingList<Foo>(Foo.class);

        public void setName(String name) {
            this.name = name;
        }

        public String getName() {
            return name;
        }

        public Collection<Foo> getList() {
            return list;
        }

        public void setList(Collection<Foo> foos) {
            this.list = foos;
        }
    }

    class Foo {
        @NotEmpty
        private String bar;

        public void setBar(String bar) {
            this.bar = bar;
        }

        public String getBar() {
            return bar;
        }
    }

    @Test
    public void testValidator() throws Exception {
        Foo foo0 = new Foo();
        foo0.setBar("");

        Foo foo1 = new Foo();
        foo1.setBar("");

        Collection<Foo> list = new ArrayList<ValidatorTest.Foo>();
        list.add(foo0);
        list.add(foo1);

        Person person = new Person();
        person.setName("Test Person");
        person.setList(list);

        Validator validator = Validation.buildDefaultValidatorFactory().getValidator();
        Set<ConstraintViolation<Person>> violations = validator.validate(person);

        for (ConstraintViolation<Person> constraintViolation : violations) {
            System.out.println(constraintViolation);
        }
    }
}

上面的测试产生以下输出:

代码语言:javascript
复制
ConstraintViolationImpl{interpolatedMessage='may not be empty', propertyPath=list[].bar, rootBeanClass=class com.foo.ValidatorTest$Person, messageTemplate='{org.hibernate.validator.constraints.NotEmpty.message}'}
ConstraintViolationImpl{interpolatedMessage='may not be empty', propertyPath=list[].bar, rootBeanClass=class com.foo.ValidatorTest$Person, messageTemplate='{org.hibernate.validator.constraints.NotEmpty.message}'}

它产生的错误是正确的;但是,propertyPath并非如此(至少在我看来是这样)。

现在,如果我将Hibernate的JSR-303实现替换为Apache的(org.apache.bval.bundle-0.2-孵化--使用标记为here的依赖项),我会得到预期的输出。这是完全相同的测试,但使用的是Apache 303注释,而不是Hibernate的。注意propertyPath字段中现在存在的索引:

代码语言:javascript
复制
ConstraintViolationImpl{rootBean=com.foo.ValidatorTest$Person@5f989f84, propertyPath='list[0].bar', message='may not be empty', leafBean=com.foo.ValidatorTest$Foo@4393722c, value=}
ConstraintViolationImpl{rootBean=com.foo.ValidatorTest$Person@5f989f84, propertyPath='list[1].bar', message='may not be empty', leafBean=com.foo.ValidatorTest$Foo@528acf6e, value=}

由于各种原因,我可能坚持使用Hibernate的JSR-303实现。我做了什么导致Hibernate验证器不填充这些索引的事情吗?我尽量避免在Spring控制器中进行手动错误处理。

感谢您能提供的任何帮助!

EN

回答 2

Stack Overflow用户

回答已采纳

发布于 2011-04-21 06:27:43

我也在Hibernate forums上问过这个问题,它在那里得到了回答。我想在这里分享答案,以防其他人遇到这个问题。

只需将Collection更改为List即可解决问题。下面是更新后的单元测试:

代码语言:javascript
复制
package com.foo;

import java.util.ArrayList;
import java.util.List;
import java.util.Set;

import javax.validation.ConstraintViolation;
import javax.validation.Valid;
import javax.validation.Validation;
import javax.validation.Validator;

import org.hibernate.validator.constraints.NotEmpty;
import org.junit.Test;
import org.springframework.util.AutoPopulatingList;

public class ValidatorTest {

    class Person {
        @NotEmpty
        private String name;

        @NotEmpty
        @Valid
        private List<Foo> list = new AutoPopulatingList<Foo>(Foo.class);

        public void setName(String name) {
            this.name = name;
        }

        public String getName() {
            return name;
        }

        public List<Foo> getList() {
            return list;
        }

        public void setList(List<Foo> foos) {
            this.list = foos;
        }
    }

    class Foo {
        @NotEmpty
        private String bar;

        public void setBar(String bar) {
            this.bar = bar;
        }

        public String getBar() {
            return bar;
        }
    }

    @Test
    public void testValidator() throws Exception {
        Foo foo0 = new Foo();
        foo0.setBar("");

        Foo foo1 = new Foo();
        foo1.setBar("");

        List<Foo> list = new ArrayList<ValidatorTest.Foo>();
        list.add(foo0);
        list.add(foo1);

        Person person = new Person();
        person.setName("Test Person");
        person.setList(list);

        Validator validator = Validation.buildDefaultValidatorFactory().getValidator();
        Set<ConstraintViolation<Person>> violations = validator.validate(person);

        for (ConstraintViolation<Person> constraintViolation : violations) {
            System.out.println(constraintViolation);
        }
    }
}

以及上述测试的输出(现在包括索引):

代码语言:javascript
复制
ConstraintViolationImpl{interpolatedMessage='may not be empty', propertyPath=list[1].bar, rootBeanClass=class com.foo.ValidatorTest$Person, messageTemplate='{org.hibernate.validator.constraints.NotEmpty.message}'}
ConstraintViolationImpl{interpolatedMessage='may not be empty', propertyPath=list[0].bar, rootBeanClass=class com.foo.ValidatorTest$Person, messageTemplate='{org.hibernate.validator.constraints.NotEmpty.message}'}
票数 5
EN

Stack Overflow用户

发布于 2021-04-17 00:00:55

如果使用对象调用方法getResultList(),但数据仍未持久保存到数据库中,则会出现相同错误。

票数 0
EN
页面原文内容由Stack Overflow提供。腾讯云小微IT领域专用引擎提供翻译支持
原文链接:

https://stackoverflow.com/questions/5724976

复制
相关文章

相似问题

领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档