12

我正在尝试为 Hibernate Validator 4.1 到 Spring 3.0 设置自定义消息源。我已经设置了必要的配置:

<!-- JSR-303 -->
<bean id="validator"
    class="org.springframework.validation.beanvalidation.LocalValidatorFactoryBean">
    <property name="validationMessageSource" ref="messageSource"/>
 </bean>

翻译是从我的消息源提供的,但似乎在消息源中查找了消息本身中的替换标记,即:

my.message=the property {prop} is invalid

有调用在 messageSource 中查找“prop”。进入ResourceBundleMessageInterpolator.interpolateMessage我注意到javadoc状态:

根据 JSR 303 中指定的算法运行消息插值。

注意:用户包中的查找是递归的,而默认包中的查找不是!

这在我看来就像递归将始终发生在用户指定的捆绑包中,所以实际上我无法翻译标准消息,如 Size 的消息。

如何插入自己的消息源并能够在消息中替换参数?

4

1 回答 1

18

这在我看来就像递归将始终发生在用户指定的捆绑包中,所以实际上我无法翻译标准消息,如 Size 的消息。

Hibernate Validator 的 ResourceBundleMessageInterpolator 创建两个 ResourceBundleLocator 实例(即 PlatformResourceBundleLocator),一个用于 UserDefined 验证消息 - userResourceBundleLocator,另一个用于 JSR-303 标准验证消息 - defaultResourceBundleLocator。

任何出现在两个大括号内的文本,例如{someText}消息中的文本都被视为replacementToken。ResourceBundleMessageInterpolator 试图找到可以替换 ResourceBundleLocators 中的replacementToken 的匹配值。

  1. 首先在 UserDefinedValidationMessages (这是递归的),
  2. 然后在 DefaultValidationMessages 中(不是递归的)。

因此,如果您将标准 JSR-303 消息放入自定义 ResourceBundle 中,例如validation_erros.properties,它将被您的自定义消息替换。请参阅此示例中的标准 NotNull 验证消息“可能不是 null”已被自定义“MyNotNullMessage”消息替换。

如何插入自己的消息源并能够在消息中替换参数?
my.message=属性 {prop} 无效

在通过两个 ResourceBundleLocators 之后,ResourceBundleMessageInterpolator 在resolvedMessage 中找到更多replaceTokens(由两个包解析)。这些replacementToken只不过是Annotation的属性名称,如果在resolvedMessage中找到这样的replaceToken,它们将被匹配的Annotation属性的值替换。

ResourceBundleMessageInterpolator.java [第 168 行,4.1.0.Final]

resolvedMessage = replaceAnnotationAttributes( resolvedMessage, annotationParameters );

提供一个用自定义值替换 {prop} 的示例,希望对您有所帮助....

MyNotNull.java

@Constraint(validatedBy = {MyNotNullValidator.class})
public @interface MyNotNull {
    String propertyName(); //Annotation Attribute Name
    String message() default "{myNotNull}";
    Class<?>[] groups() default { };
    Class<? extends Payload>[] payload() default {};
}

MyNotNullValidator.java

public class MyNotNullValidator implements ConstraintValidator<MyNotNull, Object> {
    public void initialize(MyNotNull parameters) {
    }

    public boolean isValid(Object object, ConstraintValidatorContext constraintValidatorContext) {
        return object != null;
    }
}

用户.java

class User {
    private String userName;

    /* whatever name you provide as propertyName will replace {propertyName} in resource bundle */
   // Annotation Attribute Value 
    @MyNotNull(propertyName="userName") 
    public String getUserName() {
        return userName;
    }
    public void setUserName(String userName) {
        this.userName = userName;
    }
}

validation_errors.properties

notNull={propertyName} cannot be null 

测试

public void test() {
    LocalValidatorFactoryBean factory = applicationContext.getBean("validator", LocalValidatorFactoryBean.class);
    Validator validator = factory.getValidator();
    User user = new User("James", "Bond");
    user.setUserName(null);
    Set<ConstraintViolation<User>> violations = validator.validate(user);
    for(ConstraintViolation<User> violation : violations) {
        System.out.println("Custom Message:- " + violation.getMessage());
    }
}

输出

Custom Message:- userName cannot be null
于 2010-11-23T20:53:32.193 回答