我已经实现了Long
字段和@PastOrPresent
验证的验证
public class PastOrPresentValidator implements ConstraintValidator<PastOrPresent, Long>{
@Override
public boolean isValid(Long value, ConstraintValidatorContext context) {
if (value == null) {
return false;
}
return value <= System.currentTimeMillis();
}
}
public class MyDto {
@PastOrPresent
Long timestamp;
}
出现错误:
HV000030: No validator could be found for constraint 'javax.validation.constraints.PastOrPresent' validating type 'java.lang.Long'. Check configuration for 'push.dto.timestamp'
但是,当我创建@PastOrPresentLong
之类的自定义注释并使用相同的验证器代码时,一切正常。
是否可以为javax.validation.constraints
注释添加自定义验证器?
答案 0 :(得分:0)
仅实现ConstraintValidator
接口不足以使新的验证程序正常工作。应该以某种方式注册它,以便BeanValidation提供程序知道它。这可以通过XML来完成(有关更多信息,请参见here):
<constraint-mappings
xmlns="http://xmlns.jcp.org/xml/ns/validation/mapping"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xsi:schemaLocation="http://xmlns.jcp.org/xml/ns/validation/mapping
http://xmlns.jcp.org/xml/ns/validation/mapping/validation-mapping-2.0.xsd"
version="2.0">
<!-- anything else -->
<constraint-definition annotation="javax.validation.constraints.PastOrPresent">
<validated-by include-existing-validators="true">
<value>org.mycompany.PastOrPresentLongValidator</value>
</validated-by>
</constraint-definition>
</constraint-mappings>
然后,如果您将Hibernate Validator用作BeanValidation提供程序,则还有两个选择-使用服务加载程序或编程定义。
使用服务加载程序是添加新验证程序的最简单方法。您需要做的就是创建一个META-INF/services/javax.validation.ConstraintValidator
文件,并向其添加验证程序的FQCN。请参见post下的“对非标准类使用标准约束”部分或doc中的详细示例。
使用programmatic approach,您将必须执行以下操作:
ConstraintMapping constraintMapping = configuration.createConstraintMapping();
constraintMapping
.constraintDefinition( PastOrPresent.class )
.validatedBy( PastOrPresentLongValidator.class );
然后将此约束映射添加到您的配置并从中创建一个验证器:
Validator validator = configuration.addMapping( constraintMapping )
.buildValidatorFactory()
.getValidator();
但是正如我之前提到的-Service Loader是最简单,最快的方法。