
本文深入探讨了在j*a bean validation中,当字段为`null`时,如何整合多个约束(如`@notnull`、`@length`、`@pattern`)的错误消息,并正确解析消息模板中的占位符。通过创建一个自定义的复合约束注解,结合`@reportassingleviolation`和`@overridesattribute`,可以实现一个统一且包含所有详细信息的验证失败消息,从而提升用户体验和错误诊断的准确性。
在使用J*a Bean Validation API时,常见的做法是通过组合多个注解来对字段进行多重约束,例如:
@NotNull @Length(min = 4, max = 64) @Pattern(regexp = "[A-Za-z0-9]+") String username;
当username字段的值为null时,默认情况下,只会触发@NotNull约束,并返回其默认错误消息,如“must not be null”。而@Length和@Pattern等约束通常将null视为有效输入,因此它们不会被触发,也不会贡献任何错误信息。这导致验证消息缺乏完整性,无法一次性告知用户所有未满足的条件。
尝试通过在@NotNull注解的message属性中直接拼接其他约束的消息模板,例如:
@NotNull(message = """
{jakarta.validation.constraints.NotNull.message}
AND {org.hibernate.validator.constraints.Length.message}
AND {jakarta.validation.constraints.Pattern.message}""")
@Length(min = 4, max = 64)
@Pattern(regexp = "[A-Za-z0-9]+")
String username;虽然这种方式可以将多个消息模板组合起来,但在实际的错误消息中,像{min}、{max}、{regexp}这样的占位符并不会被解析为实际的数值。这是因为这些占位符属于@Length和@Pattern约束的属性,而当前@NotNull注解本身并不拥有这些属性,因此无法提供解析所需的值。
为了解决上述问题,我们可以创建一个自定义的复合约束注解。这个注解将封装所有相关的约束,并提供一个统一的错误消息模板。关键在于使用@ReportAsSingleViolation来确保所有内部约束的验证失败都被报告为单一的违规,以及利用@OverridesAttribute来解析占位符。
首先,定义一个名为@ValidUsername的注解,它将包含@NotNull、@Length和@Pattern等约束。
import jakarta.validation.Constraint;
import jakarta.validation.Payload;
import jakarta.validation.ReportAsSingleViolation;
import jakarta.validation.constraints.NotNull;
import jakarta.validation.constraints.Pattern;
import org.hibernate.validator.constraints.Length;
import org.hibernate.validator.constraints.OverridesAttribute;
import j*a.lang.annotation.Documented;
import j*a.lang.annotation.Retention;
import j*a.lang.annotation.Target;
import static j*a.lang.annotation.ElementType.FIELD;
import static j*a.lang.annotation.RetentionPolicy.RUNTIME;
@Constraint(validatedBy = {}) // 无需显式验证器,它委托给其他约束
@NotNull // 确保字段不为null
@Length(min = 4, max = 64) // 长度约束
@Pattern(regexp = "[A-Za-z0-9]+") // 模式约束
@ReportAsSingleViolation // 将所有内部约束的违规报告为单一违规
@Target(FIELD) // 作用于字段
@Retention(RUNTIME) // 运行时保留
@Documented
public @interface ValidUsername {
String message() default """
{jakarta.validation.constraints.NotNull.message}
AND {org.hibernate.validator.constraints.Length.message}
AND {jakarta.validation.constraints.Pattern.message}""";
Class<?>[] groups() default {};
Class<? extends Payload>[] payload() default {};
}注解解析:
即使定义了复合约束,消息模板中的{min}、{max}、{regexp}等占位符仍然无法被正确解析,因为它们是@Length和@Pattern的属性,而非@ValidUsername的属性。为了解决这个问题,我们需要在@ValidUsername注解中声明这些属性,并通过@OverridesAttribute注解告诉Bean Validation框架,这些属性应该覆盖或提供给其内部的特定约束。
在@ValidUsername注解中添加以下属性:
挖错网
一款支持文本、图片、视频纠错和AIGC检测的内容审核校对平台。
185
查看详情
public @interface ValidUsername {
// ... 其他属性和方法 ...
@
OverridesAttribute(constraint = Length.class, name = "min")
int min() default 4;
@OverridesAttribute(constraint = Length.class, name = "max")
int max() default 64;
@OverridesAttribute(constraint = Pattern.class, name = "regexp")
String regexp() default "[A-Za-z0-9]+";
}@OverridesAttribute解析:
现在,您只需将字段上的所有独立约束替换为新的@ValidUsername注解:
// 替换之前的多个注解 @ValidUsername String username;
当username字段为null时,或者不满足长度、模式要求时,您将获得一个包含所有详细信息的统一错误消息,例如:
must not be null AND length must be between 4 and 64 characters AND must match "[A-Za-z0-9]+"
其中{min}、{max}和{regexp}占位符将被实际的数值和正则表达式字符串替换。
通过创建自定义复合约束注解,并结合@ReportAsSingleViolation和@OverridesAttribute,我们可以有效地解决J*a Bean Validation中多重约束消息整合和占位符解析的问题。这种方法提供了以下优势:
注意事项:
以上就是J*a Bean Validation:整合多重约束消息并解析占位符的实践的详细内容,更多请关注其它相关文章!
相关文章:
天眼查怎么看公司融资情况 天眼查企业融资历史查询步骤【攻略】
CSS实现侧边栏导航项全宽圆角悬停背景效果
sublime怎么预览Markdown渲染效果_Markdown Preview插件 for sublime教程
钉钉视频会议声音异常如何处理 钉钉会议音频修复技巧
腾讯视频怎么使用多账号家庭管理_腾讯视频家庭多账号统一管理与权限分配教程
Win11怎么设置开机NumLock亮 Win11修改注册表InitialKeyboardIndicators值
AO3官方镜像站点汇总 AO3同人作品网页版直达链接
Go语言中动态执行代码字符串的策略与实践
处理Kafka消息时会话超时与实现幂等性消费者
Yandex免登录网页版地址 Yandex搜索引擎官方访问入口
抖音网页版企业服务中心登录入口_抖音网页版企业登录平台
wps文字怎么插入目录并自动更新_wps文字如何插入目录并自动更新方法
漫蛙漫画登录站点 漫蛙2正版漫画快速访问
在J*a中如何开发简易博客标签推荐系统_博客标签推荐项目实战解析
Pandas DataFrame:高效添加条件计算列
Python实现多节点属性重叠度分析教程
Go语言中构建可靠数据存储的原子性与持久化策略
J*a应用集成GitHub CLI与API认证指南
圆通快递查询实时追踪 圆通物流包裹状态快速查看
2026年CSGO开箱网站推荐 CSGO开箱平台精选
腾讯视频怎么举报不良内容_腾讯视频内容举报流程与违规信息处理方法
优化 Jest 模拟:强制未实现函数抛出错误以提升测试效率
Win11网速慢怎么解决 Win11网络设置优化解除限速
在J*a中如何实现对象克隆避免共享数据_对象克隆安全实践指南
J*a中实现Go语言select通道多路复用机制
搜狗浏览器如何使用密码生成器创建强密码 搜狗浏览器内置密码安全工具
在Runstone环境中高效处理TasteDive API的JSON数据
Web Components中自定义开关组件状态同步的常见陷阱与解决方案
微信网页版官方入口教程 微信网页版网页版快速登录步骤
在J*aScript中复现SciPy的B样条拟合与求值:关键考量
win11跳过OOBE三种方法 Win11跳过OOBE设置步骤
qq游戏手机版下载安装_qq游戏移动端入口
Lar*el 中按“Has One Of Many”关联模型排序的最佳实践
qq游戏免费畅玩入口_qq游戏电脑版快速启动
单射、满射与双射的关系 一文理清所有逻辑
Go语言中JSON数据解码与字段访问指南
夸克浏览器网页版最新地址 夸克浏览器官方入口合集
Pyrogram与g4f集成:异步编程实践与常见错误解决
html怎么在cmd下运行php文件_cmd运行html中php文件方法【教程】
在Go语言中利用后缀数组处理多字符串:实现高效文本匹配与自动补全
MAC如何将整个网页截长图_MAC使用Safari的导出为PDF或第三方工具
Python异步编程实践:使用Binance API构建实时交易数据流
age动漫网站入口 age动漫官网直接访问入口
VS Code远程开发时如何处理文件权限问题
mc.js游戏直达 mc.js网页免下载版本秒进地址
QQ邮箱网页版入口 QQ邮箱官方邮箱登录通道
厨房不锈钢水槽发黑生锈怎么处理_水槽用可乐+锡纸2分钟抛亮如新
J*a里如何使用forEach遍历Map_Map遍历方法说明
冬*霸灯泡不亮怎么办_浴霸取暖灯一盏不亮的灯座清洁修复法
迅雷下载到U盘速度很慢怎么办_迅雷U盘下载慢优化方法