Springboot集成BeanValidation扩展一:错误提示信息加公共模板
1、需求
在使用validator时,有个需求就是公用错误提示信息,什么意思?
举个例子:
@NotEmpty非空判断,在资源文件中我不想每个非空判断都写”不能为空“,只需要写”###“,然后提示信息自动会变成”###不能为空“
代码:
public class User{ //资源文件中user.name.empty=用户名 @NotEmpty(key={user.name.empty}) private String name; ''' }
2、实现方式
有两种实现方式
方式一:手动调用验证方法
注解
@Target({FIELD, ANNOTATION_TYPE}) @Retention(RUNTIME) @ReportAsSingleViolation @Constraint(validatedBy = {}) @NotNull @Size(min = 1) public @interface NotEmpty { String message() default "{key}{com.chyjr.hyb.validator.constraints.empty.message}"; Class<?>[] groups() default { }; Class<? extends Payload>[] payload() default { }; String key() default ""; }
验证器
//验证器
public class MyValidator { private static final Logger log = LoggerFactory.getLogger(HybValidator.class); private static Validator validator = null; private static MessageInterpolator msgInterpolator = null; static { if (validator == null) { LocalValidatorFactoryBean factory = (LocalValidatorFactoryBean) ApplicationContextUtil.getBean("validator"); validator = factory.getValidator(); msgInterpolator = factory.getMessageInterpolator(); } } public static HybValidatorResult validate(Object object, Class<?>... groups) { HybValidatorResult result = new HybValidatorResult(); Set<ConstraintViolation<Object>> violations = validator.validate(object, groups); Map<String, String> map = new HashMap<>(); if (CollectionUtils.isEmpty(violations)) { result.setErrors(false); } else { result.setErrors(true); for (ConstraintViolation<Object> violation : violations) { String path = violation.getPropertyPath().toString(); String message = violation.getMessage(); if (StringUtils.isBlank(path) || StringUtils.isBlank(message) || map.containsKey(path)) continue; message = resolveMessage(message); map.put(path, message); } result.setItems(map); } return result; } private static final Pattern elpattern = Pattern.compile("\\{[^{}]+\\}"); private static String resolveMessage(String message) { Matcher matcher = elpattern.matcher(message); try { while (matcher.find()) { String el = matcher.group(); //用资源文件信息替换message = {key}{my.empty.message} //注解这里的key会替换成注解NotEmpty定义的key,即 //message = {user.name.empty}{my.empty.message} String val = msgInterpolator.interpolate(el, null); if (StringUtils.isBlank(val)) continue; message = message.replace(el, val); } } catch (Exception e) { log.error("验证引擎进行数据校验时出现异常, message:{}", message, e); } return message; } }
使用
//调用验证方法获得验证结果
HybValidatorResult bvr = HybValidator.validate(emp, CreateValidator.class); //表示有错误 if (bvr.isErrors()) { } //资源文件内容 //my.empty.message=不能为空 //user.name.empty=用户名
方式二:用spring自带的@Validated,无需调用验证方法
这里有个问题:@Validated注解不认注解@NotEmpty中的key,如何解决呢?
最终的实现方案:自定义验证器
代码:
注解
@Documented @Target({FIELD, ANNOTATION_TYPE}) @Retention(RUNTIME) @ReportAsSingleViolation //指定验证器 @Constraint(validatedBy = NotEmptyValidator.class) public @interface NotEmpty { String message() default "{my.empty.message}"; Class<?>[] groups() default { }; Class<? extends Payload>[] payload() default { }; String key() default ""; }
验证器:自定义
public class NotEmptyValidator extends AbstractValidator<NotEmpty,Object>{ @Override public void initialize(NotEmpty notEmpty) { } @Override public boolean doIsValid(Object value, ConstraintValidatorContext cc) { return value != null; } } /** * 这里采用模板的设计模式 * @param constraintAnnotation */ public abstract class AbstractValidator<A extends Annotation,T> implements ConstraintValidator<A,T>{ /** * 初始化由具体类实现 * @param constraintAnnotation */ @Override public abstract void initialize(A constraintAnnotation); /** * 初始化具体由实现类实现 * @param value * @param context * @return */ @Override public boolean isValid(T value, ConstraintValidatorContext context){ //获取验证结果,采用模板方法 boolean result = doIsValid(value,context); //当验证错误时修改默认信息 if(!result){ //改变默认提示信息 if(ConstraintValidatorContextImpl.class.isAssignableFrom(context.getClass())){ ConstraintValidatorContextImpl constraintValidatorContext = (ConstraintValidatorContextImpl)context; //获取默认提示信息 String defaultConstraintMessageTemplate = context.getDefaultConstraintMessageTemplate(); Object key = constraintValidatorContext.getConstraintDescriptor().getAttributes().get("key"); //禁用默认提示信息 context.disableDefaultConstraintViolation(); //设置提示语(在message前面加上key) context.buildConstraintViolationWithTemplate(key + defaultConstraintMessageTemplate).addConstraintViolation(); } } return result; } /** * 真正验证方法 * @param value * @param context * @return */ public abstract boolean doIsValid(T value, ConstraintValidatorContext context); }
使用:
调用的时候只要在JavaBean前加上@Validated注解即可
总结:上述就是在工作中遇到的问题,并扩展了Validator