【问题标题】:Spring + Thymeleaf custom validation displaySpring + Thymeleaf 自定义验证展示
【发布时间】:2015-12-13 05:09:03
【问题描述】:

我一直在尝试使自定义 javax 验证正常工作(Spring Boot 和 Thymeleaf),但我不知道如何显示错误消息。问题似乎是,“正常”错误(例如@Size、@NotNull 等)似乎在绑定结果中添加了一个 FieldError。我的自定义验证器虽然提供了一个 ObjectError。我不知道如何让 Thymeleaf 显示我的自定义错误(显然已通过,因为 th:errors="*{*}" 显示了它)。

非常感谢任何帮助。

更新: 我现在可以通过

显示错误消息
<p th:if="${#fields.hasErrors('${user}')}" th:errors="${user}"></p>

但是,如果我需要多个验证器(例如确认密码和确认电子邮件),此解决方案将不起作用(或者如果一个不适合,则显示两条错误消息。如果您有建议,请不要犹豫。

以下是我为此使用的代码:

模板:

<p th:if="${#fields.hasErrors('username')}"th:errors="*{username}"></p>
<!-- works just fine -->
<p th:if="${#fields.hasErrors('*')}" th:errors="*{*}"></p>
<!-- works and displays all errors (for every field with an error,
 including my custom validator) -->
<p th:if="${#fields.hasErrors('confirmPassword')}" th:errors="*{*}"></p>
<!-- does not work -->
<p th:if="${#fields.hasErrors('*')}" th:errors="*{confirmPassword}"></p>
<!-- does not work -->

验证器实现:

public class PasswordsEqualConstraintValidator implements
        ConstraintValidator<PasswordsEqualConstraint, Object> {

    @Override
    public void initialize(PasswordsEqualConstraint arg0) {
    }

    @Override
    public boolean isValid(Object candidate, ConstraintValidatorContext arg1) {
        User user = (User) candidate;
        return user.getPassword().equals(user.getConfirmPassword());
    }
}

验证器接口:

@Target({ ElementType.TYPE })
@Retention(RetentionPolicy.RUNTIME)
@Documented
@Constraint(validatedBy = PasswordsEqualConstraintValidator.class)
public @interface PasswordsEqualConstraint {
    String message();

    Class<?>[] groups() default {};

    Class<? extends Payload>[] payload() default {};
}

用户.java:

@PasswordsEqualConstraint(message = "passwords are not equal")
public class User implements java.io.Serializable {
...     
@Size(min=2, max=40)
private String username;
...
private String confirmPassword;
...

控制器:

@RequestMapping(value = "/signup", method = RequestMethod.POST)
public String signup(@Valid User user, BindingResult bindingResult) {
        if (bindingResult.hasErrors()) {
            return "signup";
        }
        ... do db stuff .. return "success";
}

【问题讨论】:

    标签: java spring validation spring-mvc thymeleaf


    【解决方案1】:

    这可能是因为您的 @PasswordsEqualConstraint 被分配给整个 bean(类型)而不是字段“confirmPassword”。要将可能的约束违规添加到具体字段,您可以像下面的示例一样。

    FieldMatch 比较两个字段,如果它们不相等,则将验证错误分配给第二个字段。

    顺便说一句。这是您正在做的事情的更通用的解决方案。例如,您可以像

    一样使用密码
    @FieldMatch(first = "password", second = "confirmPassword", message = "Passowords are not equal.")
    

    验证者:

    public class FieldMatchValidator implements ConstraintValidator<FieldMatch, Object> {
    
      private String firstFieldName;
      private String secondFieldName;
    
      @Override
      public void initialize(final FieldMatch constraintAnnotation) {
        firstFieldName = constraintAnnotation.first();
        secondFieldName = constraintAnnotation.second();
      }
    
      @Override
      public boolean isValid(final Object value, final ConstraintValidatorContext context) {
        try {
          final Object firstObj = BeanUtils.getProperty(value, firstFieldName);
          final Object secondObj = BeanUtils.getProperty(value, secondFieldName);
    
          boolean isValid = firstObj == null && secondObj == null || firstObj != null && firstObj.equals(secondObj);
    
          if (!isValid) {
            context.disableDefaultConstraintViolation();
            context.buildConstraintViolationWithTemplate(context.getDefaultConstraintMessageTemplate()).addNode(secondFieldName).addConstraintViolation();
          }
    
          return isValid;
        }
        catch (final Exception ignore) {
          // ignore
        }
        return true;
      }
    }
    

    【讨论】:

    • 像魅力一样工作,非常感谢!您的答案的一个小改动是使用 addPropertyNode() 而不是已弃用的 addNode() 方法。
    【解决方案2】:

    我终于得到了批准的答案,但只是在从第一个问题部分获得详细信息之后。起初并不清楚如何定义 @FieldMatch 接口。主要是这样

    @Target({ElementType.TYPE})
    @Retention(RetentionPolicy.RUNTIME)
    @Constraint(validatedBy = FieldMatchValidator.class)
    

    在 FieldMatch 界面中是必需的。

    这里是 spring 文档。 http://dolszewski.com/spring/custom-validation-annotation-in-spring/

    spring 文档显示 @Target({ElementType.METHOD, ElementType.FIELD})

    这让我很困惑,因为当 isValid 被调用时,目标是 Method 和 Field,Object 只是我放置 @FieldMatch 注释的 1 属性的值。但是拥有 @Target({ElementType.TYPE}) 然后当 Valid 被调用时,我得到了正在验证的整个 bean,我可以对其进行强制转换或使用反射来获取值。

    【讨论】:

      猜你喜欢
      • 2018-05-07
      • 1970-01-01
      • 2013-05-29
      • 2011-06-15
      • 1970-01-01
      • 2014-09-24
      • 2011-05-01
      • 2017-07-14
      • 2018-01-29
      相关资源
      最近更新 更多