一尘不染

如何使Hibernate Validator在第一次违反字段时停止验证?

hibernate

我有一个bean,我在其中为每个字段定义多个验证注释,例如

@NotEmpty
@Pattern(regexp="(\\-?\\d)+")
@Min(value=1)
String myField;

我遇到了 两个 无法以任何简单方式解决的问题。

  1. 每个字段的指定批注的验证顺序是随机的,即不会按照定义批注的顺序进行。 我相信@GroupSequence将无济于事,因为它仅定义组验证序列,而不是注释序列。 正如@Tom正确注释的那样,违规被报告为Set,这意味着注释的执行顺序与所报告的违规之间没有1:1的映射。
  2. 我只想使每个字段的一条规则无效,即,如果它与模式不匹配,请不要尝试检查该值是否> =1。当前,如果将myField设置为“ abc”,它将报告@Pattern和@Min违例。将验证器的failFast属性设置为true并没有帮助,因为我使用同一验证器实例来验证bean中的所有字段,并且在遇到整个bean的第一个违规时,它将停止验证其他字段。

编辑。我尝试使用@ReportAsSingleViolation实现自定义复合约束。问题在于,它将为撰写中涉及的所有违规报告相同的消息。这不是我所需要的。

建议,好吗?


阅读 311

收藏
2020-06-20

共1个答案

一尘不染

应该可以结合使用验证组和完全定义的验证组序列来创建想要实现的行为

引用JSR-349(aka.BeanValidation
1.1)规范,第4.4.2段

第4.6节中 定义了处理组; 如果序列中处理的组之一产生一个或多个约束冲突,则 不得处理 序列中的后续组
这样可以确保仅在另一组约束有效时才评估一组约束。
由我突出

通过组序列示例,这应该相对容易一些

@GroupSequence({YourClass.class, Second.class, Third.class})
public class YourClass {
    @NotNull
    @Pattern(regexp="(\\-?\\d)+", groups=Second.class)
    @Min(value=1, groups=Third.class)
    String myField;
}

SecondThird定义为简单的标记接口应该做的伎俩。请注意,这不会验证所有字段,直到第一个约束违反,而是直到第一个总体违反。

这意味着你的一些领域可能会 成为 固定等领域的违规行为后无效。

如果所有其他方法都失败了,您仍然可以重新实现行为组件,即验证提供程序。您应该能够ConstraintValidatorContext通过使用validate.xml
提供自己的实现

请注意,您 必须
违约ConstraintValidatorContext.ConstraintViolationBuilder

要创建ConstraintViolation,必须调用fluent API的一个接口中可用的addConstraintViolation()方法之一。

如果在ConstraintViolationBuilder或其任何关联对象的 addConstraintViolation()之后
调用了另一个方法, IllegalStateException则会引发。

2020-06-20