我正在尝试使用Spring MVC设置JSR-303表单验证。我已经正确配置了所有内容(或者至少我认为我这样做了),并且验证工作正常。但是,如果我有一个包含我想要验证的对象集合的命令对象,并且我使用@Valid注释该集合,则Hibernate JSR-303提供程序不提供正确的propertyPath。 ContraintViolation对象中的propertyPath应该填充list[0].bar
和list[1].bar
,但Hibernate验证器只提供list[].bar
和list[].bar
。当Spring的SpringValidatorAdaptor.validate()方法尝试添加字段级错误时,这会导致NumberFormatException(因为它在内部期望这些括号中存在一个数字)。
使用spring-context-3.0.5和hibernate-validator-4.1.0.Final(我也试过4.0.2.GA和4.2.0.Beta2得到了相同的结果),我写了一个小单元测试说明了这个问题:
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);
}
}
}
上述测试产生以下输出:
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不是(至少根据我的理解)。
现在,如果我用Apache替换Hibernate的JSR-303实现(org.apache.bval.bundle-0.2-incubating - 使用标记为here的依赖项),我得到了我期望的输出。这是完全相同的测试,但使用Apache的JSR-303注释而不是Hibernate。请注意propertyPath字段中现在存在的索引:
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控制器中进行手动错误处理。
感谢您提供的任何帮助!
答案 0 :(得分:5)
我也在Hibernate forums上问了这个问题,并在那里得到了回答。我想在这里分享答案,以防其他人遇到这个问题。
只需将Collection更改为List即可解决问题。这是更新的单元测试:
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);
}
}
}
以上测试的输出(现在包括索引):
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}'}
答案 1 :(得分:0)
如果您使用对象调用方法 getResultList() 时出现相同的错误,而该对象的数据仍未保存到数据库中。