Bean Validator扩展

1、需求

​ 在使用validator时,有个需求就是公用错误提示信息,什么意思?

举个例子:

​ @NotEmpty非空判断,在资源文件中我不想每个非空判断都写”不能为空“,只需要写”###“,然后提示信息自动会变成”###不能为空“

代码:

public class User{
//资源文件中user.name.empty=用户名
@NotEmpty(key={user.name.empty})
private String name;
'''
}
//加入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

Springboot集成BeanValidation扩展一:错误提示信息加公共模板的更多相关文章

  1. Springboot集成BeanValidation扩展二:加载jar中的资源文件

    一.需求 今天在搭建Springboot框架的时候,又遇到一个需求:在多模块系统中,有些模块想自己管理BeanValidation的资源文件(默认是启动项目claspath下的 ValidationM ...

  2. Springboot 集成 Thymeleaf 及常见错误

    Thymeleaf模板引擎是springboot中默认配置,与freemarker相似,可以完全取代jsp,在springboot中,它的默认路径是src/main/resources/templat ...

  3. springboot集成mybatis源码分析-启动加载mybatis过程(二)

    1.springboot项目最核心的就是自动加载配置,该功能则依赖的是一个注解@SpringBootApplication中的@EnableAutoConfiguration 2.EnableAuto ...

  4. springboot 集成apollo,根据不同环境加载配置

  5. SpringBoot集成Shiro 实现动态加载权限

    一.前言 本文小编将基于 SpringBoot 集成 Shiro 实现动态uri权限,由前端vue在页面配置uri,Java后端动态刷新权限,不用重启项目,以及在页面分配给用户 角色 . 按钮 .ur ...

  6. Springboot学习04-默认错误页面加载机制源码分析

    Springboot学习04-默认错误页面加载机制源码分析 前沿 希望通过本文的学习,对错误页面的加载机制有这更神的理解 正文 1-Springboot错误页面展示 2-Springboot默认错误处 ...

  7. SpringBoot集成Swagger2实现Restful(类型转换错误解决办法)

    1.pom.xml增加依赖包 <dependency> <groupId>io.springfox</groupId> <artifactId>spri ...

  8. springboot集成websocket实现向前端浏览器发送一个对象,发送消息操作手动触发

    工作中有这样一个需示,我们把项目中用到代码缓存到前端浏览器IndexedDB里面,当系统管理员在后台对代码进行变动操作时我们要更新前端缓存中的代码怎么做开始用想用版本方式来处理,但这样的话每次使用代码 ...

  9. SpringBoot中BeanValidation数据校验与优雅处理详解

    目录 本篇要点 后端参数校验的必要性 不使用Validator的参数处理逻辑 Validator框架提供的便利 SpringBoot自动配置ValidationAutoConfiguration Va ...

随机推荐

  1. 利用traceback精确定位错误发生的位置

    背景:在线上代码发生bug时经常只知道错误的原因,但是很难快速的定位到错误发生的位置. 如下图,我们只知道错误. 而在try...except...后添加traceback即可以明确的抛出错误的地址. ...

  2. FormsAuthentication类

    理解代码: string cookieName = FormsAuthentication.FormsCookieName; FormsAuthentication类说明: // 摘要: // 为 W ...

  3. 哈夫曼树(C++优先队列的使用)

       给定n个权值作为n个叶子结点,构造一棵二叉树,若带权路径长度达到最小,称为哈夫曼树(Huffman Tree).哈夫曼树是带权路径长度最短的树,权值较大的结点离根较近.    构造 假设有n个权 ...

  4. hdu2010(dfs+剪枝)

    Tempter of the Bone Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 65536/32768 K (Java/Othe ...

  5. HCNA--tcp/ip基础

    1.OSI七层参考模型 应用层 7 表示层 6  高层:负责主机之间的的数据传输 会话层 5 传输层 4 主机之间端到端的连接 tcp /udp segment 网络层 3 路由 寻址  packet ...

  6. Tomcat的context.xml说明、Context标签讲解

    Tomcat的context.xml说明.Context标签讲解 1. 在tomcat 5.5之前 --------------------------- Context体现在/conf/server ...

  7. POJ3207 Ikki's Story IV – Panda's Trick

    Time Limit: 1000MS   Memory Limit: 131072K Total Submissions: 9426   Accepted: 3465 Description liym ...

  8. MySql实现分页查询的SQL,mysql实现分页查询的sql语句(转)

    http://blog.csdn.net/sxdtzhaoxinguo/article/details/51481430 摘要:MySQL数据库实现分页查询的SQL语句写法! 一:分页需求: 客户端通 ...

  9. 使用windos模拟搭建web集群(二)

    一.通过rsync搭建备份服务器 这三个目录我们需要做实时热备,他们分别是  系统的脚本目录  系统的配置文件目录  系统的定时任务目录 [root@mage-monitor- ~]# cat /se ...

  10. 爬虫练习二:GUI+下载百思不得姐网站视频

    环境 python2.7 pycharm 课题:Python爬取视频(桌面版)---爬虫,桌面程序应用 优点:语法简洁,入门快,代码少,开发效率高,第三方库 1.图形用户界面---GUI 2.爬虫,爬 ...