扩展SSM的验证注释,使其能使用资源文件及动态参数返回错误信息

本文主要是介绍扩展SSM的验证注释,使其能使用资源文件及动态参数返回错误信息,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

1/ POM.xml增加两个依赖

<!-- https://mvnrepository.com/artifact/javax.validation/validation-api --><dependency><groupId>javax.validation</groupId><artifactId>validation-api</artifactId><version>2.0.1.Final</version></dependency><!-- https://mvnrepository.com/artifact/org.hibernate.validator/hibernate-validator --><dependency><groupId>org.hibernate.validator</groupId><artifactId>hibernate-validator</artifactId><version>6.0.16.Final</version></dependency>

2/跟踪Spring MVC的代码发现, 当验证失败后,通过MethodArgumentNotValidException来抛出异常, 那么只要我们拦截这个异常, 然后将里面的错误信息翻译一下即可, 我做的一个拦截器, message里面支持下面几种写法:
       a/兼容原先的直接写错误信息
       b/支持用{}符号调用资源文件来显示错误信息
       c/在b的基础上,带入参数, 参数可以为任何常量, 又或者以#{target和#{self开头的属性. target指的是当前验证的类的实例, self指的是当前的注释

比如:

如果参数类的str="abc123526262356233562",则

返回的错误效果:

 

拦截器也简单, 就是在原先全局异常处理里面再增加分析MethodArgumentNotValidException:

	/***** 全局异常处理:全部以Json格式返回至前端, 并且在日志中打印调用堆栈* * @param e* @return*/@ExceptionHandler(value = Exception.class)public @ResponseBody Object handleException(Exception e) {if (!(e instanceof ModifyDataException || e instanceof MethodArgumentNotValidException)) {e.printStackTrace();}if (e instanceof MethodArgumentNotValidException) {MethodArgumentNotValidException lvExp = (MethodArgumentNotValidException) e;Object lvTarget = lvExp.getBindingResult().getTarget();// Method lvMethod=lvExp.getParameter().getMethod();StringBuilder sbMsg = new StringBuilder();StringBuilder sbRefs = new StringBuilder();for (ObjectError item : lvExp.getBindingResult().getAllErrors()) {FieldError lvError = (FieldError) item;String lvsDefaultMsg = lvError.getDefaultMessage();if (lvsDefaultMsg.startsWith("{")) {if (!lvsDefaultMsg.endsWith("}")) {throw new RuntimeException("miss '}' with error Message pattern:" + lvsDefaultMsg);}String lvsTmp = StringUtils.left(lvsDefaultMsg, lvsDefaultMsg.length() - 1);String[] lvItems = StringUtils.right(lvsTmp, lvsTmp.length() - 1).split(",");if (lvItems.length == 0 || lvItems.length == 1 && lvItems[0].isEmpty()) {throw new RuntimeException("error Message pattern is blank");}String lvsKey = lvItems[0];List<Object> lvParams = new ArrayList<Object>();for (int i = 1; i < lvItems.length; i++) {String lvsParam = lvItems[i];if (lvsParam.startsWith("#{") && lvsParam.endsWith("}")) { // #{target.email}lvsTmp = StringUtils.left(lvsParam, lvsParam.length() - 1);lvsTmp = StringUtils.right(lvsTmp, lvsTmp.length() - 2);String[] lvWd = lvsTmp.split("\\.");String lvsInst = lvWd[0];String lvsField = lvWd[1];Object lvInst = null;if (lvsInst.equals("target")) {lvInst = lvTarget;if (lvInst == null) {throw new RuntimeException("Invalid error message param:'" + lvsParam + "'");}Object lvParamVal = BeanUtil.getFieldValue(lvInst, lvsField, Object.class);lvParams.add(lvParamVal);} else if (lvsInst.equals("self")) {try {Field lvField = lvTarget.getClass().getDeclaredField(lvError.getField());String lvsAnnName = lvError.getCodes()[2];Annotation lvAnn = null;for (Annotation ann : lvField.getAnnotations()) {String lvsClName = ann.annotationType().getSimpleName();if (lvsClName.equals(lvsAnnName)) {lvAnn = ann;break;}}if (lvAnn == null) {throw new RuntimeException("Invalid error message param:'" + lvsParam + "'");}Map<String, Object> lvAttrs = AnnotationUtils.getAnnotationAttributes(lvAnn);Object lvParamVal = lvAttrs.get(lvsField);lvParams.add(lvParamVal);} catch (NoSuchFieldException | SecurityException e1) {throw new RuntimeException(e1);}}} else {lvParams.add(lvsParam);}}sbMsg.append("\n" + MessageSourceHelper.getMessage(lvsKey,lvParams.toArray(new Object[lvParams.size()]), "<" + lvsKey + ">", Locale.CHINA));} else {sbMsg.append("\n" + lvsDefaultMsg);}String lvFieldName = lvError.getField();@SuppressWarnings("rawtypes")Class lvCz = lvExp.getBindingResult().getTarget().getClass();Field lvF;try {lvF = lvCz.getField(lvFieldName);Column lvCol = lvF.getAnnotation(Column.class);if (lvCol != null) {lvFieldName = lvCol.name();}sbRefs.append("," + lvFieldName);} catch (NoSuchFieldException | SecurityException e1) {// TODO Auto-generated catch blocke1.printStackTrace();}}if (sbRefs.length() > 0) {sbRefs.delete(0, 1);}if (sbMsg.length() > 0) {sbMsg.delete(0, 1);}e = new ModifyDataException(sbMsg.toString(), sbRefs.toString().split(","));}return new JSONData(-1, e);}

 

这篇关于扩展SSM的验证注释,使其能使用资源文件及动态参数返回错误信息的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/189875

相关文章

使用Java解析JSON数据并提取特定字段的实现步骤(以提取mailNo为例)

《使用Java解析JSON数据并提取特定字段的实现步骤(以提取mailNo为例)》在现代软件开发中,处理JSON数据是一项非常常见的任务,无论是从API接口获取数据,还是将数据存储为JSON格式,解析... 目录1. 背景介绍1.1 jsON简介1.2 实际案例2. 准备工作2.1 环境搭建2.1.1 添加

如何使用celery进行异步处理和定时任务(django)

《如何使用celery进行异步处理和定时任务(django)》文章介绍了Celery的基本概念、安装方法、如何使用Celery进行异步任务处理以及如何设置定时任务,通过Celery,可以在Web应用中... 目录一、celery的作用二、安装celery三、使用celery 异步执行任务四、使用celery

使用Python绘制蛇年春节祝福艺术图

《使用Python绘制蛇年春节祝福艺术图》:本文主要介绍如何使用Python的Matplotlib库绘制一幅富有创意的“蛇年有福”艺术图,这幅图结合了数字,蛇形,花朵等装饰,需要的可以参考下... 目录1. 绘图的基本概念2. 准备工作3. 实现代码解析3.1 设置绘图画布3.2 绘制数字“2025”3.3

Jsoncpp的安装与使用方式

《Jsoncpp的安装与使用方式》JsonCpp是一个用于解析和生成JSON数据的C++库,它支持解析JSON文件或字符串到C++对象,以及将C++对象序列化回JSON格式,安装JsonCpp可以通过... 目录安装jsoncppJsoncpp的使用Value类构造函数检测保存的数据类型提取数据对json数

python使用watchdog实现文件资源监控

《python使用watchdog实现文件资源监控》watchdog支持跨平台文件资源监控,可以检测指定文件夹下文件及文件夹变动,下面我们来看看Python如何使用watchdog实现文件资源监控吧... python文件监控库watchdogs简介随着Python在各种应用领域中的广泛使用,其生态环境也

Python中构建终端应用界面利器Blessed模块的使用

《Python中构建终端应用界面利器Blessed模块的使用》Blessed库作为一个轻量级且功能强大的解决方案,开始在开发者中赢得口碑,今天,我们就一起来探索一下它是如何让终端UI开发变得轻松而高... 目录一、安装与配置:简单、快速、无障碍二、基本功能:从彩色文本到动态交互1. 显示基本内容2. 创建链

springboot整合 xxl-job及使用步骤

《springboot整合xxl-job及使用步骤》XXL-JOB是一个分布式任务调度平台,用于解决分布式系统中的任务调度和管理问题,文章详细介绍了XXL-JOB的架构,包括调度中心、执行器和Web... 目录一、xxl-job是什么二、使用步骤1. 下载并运行管理端代码2. 访问管理页面,确认是否启动成功

使用Nginx来共享文件的详细教程

《使用Nginx来共享文件的详细教程》有时我们想共享电脑上的某些文件,一个比较方便的做法是,开一个HTTP服务,指向文件所在的目录,这次我们用nginx来实现这个需求,本文将通过代码示例一步步教你使用... 在本教程中,我们将向您展示如何使用开源 Web 服务器 Nginx 设置文件共享服务器步骤 0 —

Java中switch-case结构的使用方法举例详解

《Java中switch-case结构的使用方法举例详解》:本文主要介绍Java中switch-case结构使用的相关资料,switch-case结构是Java中处理多个分支条件的一种有效方式,它... 目录前言一、switch-case结构的基本语法二、使用示例三、注意事项四、总结前言对于Java初学者

Golang使用minio替代文件系统的实战教程

《Golang使用minio替代文件系统的实战教程》本文讨论项目开发中直接文件系统的限制或不足,接着介绍Minio对象存储的优势,同时给出Golang的实际示例代码,包括初始化客户端、读取minio对... 目录文件系统 vs Minio文件系统不足:对象存储:miniogolang连接Minio配置Min