Hibernate-Validator扩展之自定义注解
一、Hibernate-Validator介绍
Hibernate-Validator框架提供了一系列的注解去校验字段是否符合预期,如@NotNull注解可以校验字段是否为null,如果为null则抛出对应的异常提示信息,通过注解大大减少了我们日常的开发工作量。包括流行的spring-boot-starter-validation,底层也是靠Hibernate-Validator实现的。
但是在实际的开发中,现有的注解可能不能满足我们的校验需求,Hibernate-Validator框架就贴心的提供了扩展,通过自定义校验注解来封装我们自己的校验逻辑。
二、自定义校验注解
下面以一个例子去说明如何根据自己的业务需求,去自定义校验注解。
需求背景:
在日常开发中,我们经常需要在Controller接口对入参的字段做校验,而且有些字段的值只允许在某个枚举定义范围内,如果不在枚举范围内,则抛出异常和错误信息。针对这种情况,我们可以自定义一个注解@StatusCodeCheck去实现。
根据官方文档的描述,自定义校验注解需要如下三个步骤:
- 创建一个约束注解
- 实现一个校验器
- 定义默认的错误信息
2.1创建一个约束注解
创建之前可以先看看@NotNull的源码
@Target({ElementType.METHOD, ElementType.FIELD, ElementType.ANNOTATION_TYPE, ElementType.CONSTRUCTOR, ElementType.PARAMETER, ElementType.TYPE_USE})
@Retention(RetentionPolicy.RUNTIME)
@Repeatable(NotNull.List.class)
@Documented
@Constraint(
validatedBy = {}
)
public @interface NotNull {
String message() default "{javax.validation.constraints.NotNull.message}";
Class<?>[] groups() default {};
Class<? extends Payload>[] payload() default {};
@Target({ElementType.METHOD, ElementType.FIELD, ElementType.ANNOTATION_TYPE, ElementType.CONSTRUCTOR, ElementType.PARAMETER, ElementType.TYPE_USE})
@Retention(RetentionPolicy.RUNTIME)
@Documented
public @interface List {
NotNull[] value();
}
}
@Target: 该元注解用于指定注解的作用范围,包括类,方法,字段等。
@Retention: 指定该元注解的保留策略,包括source,class和runtime。
@Repeatable: 这是一个很有意思的元注解,在没有@Repeatable注解的的注解中,在同一个地方使用相同的注解会报错,有了此元注解注解的注解,就可以在同一个地方使用相同的注解。
@Documented: 表示会生成Java doc
@Constraint: 用于指定校验器,通过校验器返回的结果(true/false)来判断是否抛出异常信息。
除了这些元注解,还有一些属性,其作用写在了注释中
package cn.sp.validation;
import javax.validation.Constraint;
import javax.validation.Payload;
import java.lang.annotation.*;
/**
* @author Ship
* @version 1.0.0
* @description:
* @date 2021/11/10 16:52
*/
@Target({ElementType.FIELD, ElementType.PARAMETER})
@Retention(RetentionPolicy.RUNTIME)
@Constraint(validatedBy = StatusCodeCheckValidator.class)
@Documented
public @interface StatusCodeCheck {
// 指定校验失败时的异常信息,后面会详细说明
String message() default "{cn.sp.validation.StatusCode.message}";
// 分组,如同一个实体类的字段有些情况需要该校验,有些情况不需要,则可通过指定分组实现
Class<?>[] groups() default {};
// 指定错误的级别,一般不会用
Class<? extends Payload>[] payload() default {};
// 自定义的属性
Class<? extends StatusCode> value();
}
StatusCode是一个泛型的接口,所以需要使用@StatusCodeCheck注解的枚举都需要实现StatusCode接口,主要起一个标记作用。
public interface StatusCode<T> {
/**
* 获取code
*
* @return
*/
T getCode();
}
2.2 实现一个校验器
实现一个校验器很简单,创建一个类实现ConstraintValidator接口即可,ConstraintValidator的源码如下
public interface ConstraintValidator<A extends Annotation, T> {
default void initialize(A constraintAnnotation) {
}
boolean isValid(T var1, ConstraintValidatorContext var2);
}
该接口是一个泛型接口,A表示作用的注解,T表示被校验对象的类型,里面有两个方法需要实现。
initialize(A constraintAnnotation)
校验器的初始化逻辑,一般用于获取自定义注解的属性,该方法是可选的。
isValid(T var1, ConstraintValidatorContext var2)
该方法有两个参数,var1为被校验的对象,var2是一个上下文提供了很多API去操作默认约束信息等,返回值表示校验是否通过,即真正的校验逻辑处理都在该方法中完成。
知道这些后,就可以开始写自己的校验器StatusCodeCheckValidator了
/**
* @author Ship
* @version 1.0.0
* @description:
* @date 2021/11/10 17:06
*/
public class StatusCodeCheckValidator implements ConstraintValidator<StatusCodeCheck, Object> {
private Class<? extends StatusCode> enumClass;
/**
* 枚举缓存
*/
private static final Map<Class<? extends StatusCode>, List<StatusCode>> CACHE_MAP = new ConcurrentHashMap<>(64);
@Override
public void initialize(StatusCodeCheck constraintAnnotation) {
this.enumClass = constraintAnnotation.value();
}
@Override
public boolean isValid(Object object, ConstraintValidatorContext constraintValidatorContext) {
if (object == null) {
return false;
}
if (!enumClass.isEnum()) {
throw new RuntimeException("StatusCode 的实现类必须是枚举类型");
}
List<StatusCode> statusCodeList = CACHE_MAP.computeIfAbsent(enumClass, (key) -> {
try {
Method method = key.getDeclaredMethod("values");
StatusCode[] statusCodes = (StatusCode[]) method.invoke(null);
return Stream.of(statusCodes).collect(Collectors.toList());
} catch (Exception e) {
e.printStackTrace();
}
return Lists.newArrayList();
});
for (StatusCode statusCode : statusCodeList) {
if (statusCode.getCode().equals(object)) {
return true;
}
}
return false;
}
}
这里通过反射来获取所有的枚举实例,后面发现用 EnumSet.of() 方法也是可以的,出于好奇就看了下它的源码,发现底层也是通过反射调用values方法+缓存来实现的,这就叫万变不离其宗吧。
2.3定义默认的错误信息
@StatusCodeCheck注解的message属性可以指定默认错误信息,既用可以写死字符串的方式如
String message() default "error message";
也可以通过${}符号去读取ValidationMessages.properties文件配置的信息
String message() default "{javax.validation.constraints.NotNull.message}";
ValidationMessages.properties
cn.sp.validation.StatusCode.message=can not find code in {value}.
这里的{value}会读取@StatusCodeCheck注解的value,功能还是挺强大的。
三、测试
首先,编写测试代码,创建用于测试的枚举类ThirdPartyPlatformEnum
/**
* @author Ship
* @version 1.0.0
* @description
* @date 2021/11/02 11:25
*/
public enum ThirdPartyPlatformEnum implements StatusCode<String> {
/**
* 拼多多
*/
PDD("PDD", "拼多多"),
/**
* 天猫
*/
TIAN_MALL("TIAN_MALL", "天猫"),
/**
* 有赞
*/
YOU_ZAN("YOU_ZAN", "有赞"),
/**
* 美团
*/
MEI_TUAN("MEI_TUAN", "美团");
private String code;
private String desc;
ThirdPartyPlatformEnum(String code, String desc) {
this.code = code;
this.desc = desc;
}
@Override
public String getCode() {
return code;
}
public String getDesc() {
return desc;
}
}
测试实体类ValidationTest
public class ValidationTest {
@StatusCodeCheck(message = "无效的第三方平台类型", value = ThirdPartyPlatformEnum.class)
private String thirdPartyPlatform;
public String getThirdPartyPlatform() {
return thirdPartyPlatform;
}
public void setThirdPartyPlatform(String thirdPartyPlatform) {
this.thirdPartyPlatform = thirdPartyPlatform;
}
}
测试接口
@RequestMapping("/validation")
@RestController
public class ValidationTestTestController {
@PostMapping("/test")
public void test(@RequestBody @Validated ValidationTest validationTest) {
System.out.println("validation test");
}
}
然后启动项目,请求接口http://localhost:9001/validation/test,请求参数如下
{
"thirdPartyPlatform":"ali"
}
控制台日志显示校验未通过,因为"ali"不在ThirdPartyPlatformEnum的code范围内。
Resolved [org.springframework.web.bind.MethodArgumentNotValidException: Validation failed for argument [0] in public void cn.sp.validation.test.ValidationTestTestController.test(cn.sp.validation.ValidationTest): [Field error in object 'validationTest' on field 'thirdPartyPlatform': rejected value [ali]; codes [StatusCodeCheck.validationTest.thirdPartyPlatform,StatusCodeCheck.thirdPartyPlatform,StatusCodeCheck.java.lang.String,StatusCodeCheck]; arguments [org.springframework.context.support.DefaultMessageSourceResolvable: codes [validationTest.thirdPartyPlatform,thirdPartyPlatform]; arguments []; default message [thirdPartyPlatform],class cn.sp.validation.ThirdPartyPlatformEnum]; default message [无效的第三方平台类型]] ]
改为PDD再次请求
{
"thirdPartyPlatform":"PDD"
}
发现控制台打印出了validation test,说明校验通过。
四、总结
Hibernate-Validator框架如何实现可以自定义注解的原理还需要深入研究下,同时在阅读英文官方文档时,感觉自己的英语水平还是不够啊。本文代码已经上传到github,如果有兴趣可以自行下载。
参考:
https://docs.jboss.org/hibernate/stable/validator/reference/en-US/html_single/#preface
Hibernate-Validator扩展之自定义注解的更多相关文章
- Hibernate validator使用和自定义validator及整合Spring MVC
http://blog.csdn.net/lwphk/article/details/43983669 Hibernate validator使用 导入validation-api-xxx.jar 以 ...
- 用spring的@Validated注解和org.hibernate.validator.constraints.*的一些注解在后台完成数据校验
这个demo主要是让spring的@Validated注解和hibernate支持JSR数据校验的一些注解结合起来,完成数据校验.这个demo用的是springboot. 首先domain对象Foo的 ...
- hibernate validator参数校验&自定义校验注解
参数校验:简单的就逐个手动写代码校验,推荐用Valid,使用hibernate-validator提供的,如果参数不能通过校验,报400错误,请求格式不正确: 步骤1:在参数对象的属性上添加校验注解如 ...
- Hibernate Validator注解大全
hibernate Validator 是 Bean Validation 的参考实现 .Hibernate Validator 提供了 JSR 303 规范中所有内置 constraint 的实现, ...
- SpringBoot Validation参数校验 详解自定义注解规则和分组校验
前言 Hibernate Validator 是 Bean Validation 的参考实现 .Hibernate Validator 提供了 JSR 303 规范中所有内置 constraint 的 ...
- hibernate validator【原】
hibernate validator 功能 在开发中经常做一些字段校验的功能,比如非空,长度限制,邮箱验证等等,为了省掉这种冗长繁琐的操作,hibernate validator提供了一套精简的注释 ...
- SpringMVC--使用hibernate validator数据校验
JSR 303 Spring3开始支持JSR 303 验证框架,JSR303是Java为Bean数据合法性校验所提供的标准框架.JSR 303 支持XML和注解风格的验证,通过在Bean属性上标注类似 ...
- 使用Hibernate Validator来帮你做数据校验
数据校验是贯穿所有应用程序层(从表示层到持久层)的常见任务.通常在每个层中实现相同的验证逻辑,这是耗时且容易出错的.这里我们可以使用Hibernate Validator来帮助我处理这项任务.对此,H ...
- hibernate validator自定义校验注解以及基于服务(服务组)的校验
hibernate validator是Bean Validation 1.1 (JSR 349) Reference Implementation,其广泛的应用在mvc的参数校验中,尤其是使用服务端 ...
- hibernate validation内置注解及自定义注解
Bean Validation 中内置的 constraint @Null 被注释的元素必须为 null @NotNull 被注释的元素必须不为 null @AssertTrue 被注释的元素必须为 ...
随机推荐
- Concat、Push、Spread syntax性能差异对比
今天在力扣上做了一道数组扁平化的题,按理来说,应该熟能生巧了,但是在使用concat时候超出了时间限制,使用push可以通过,代码如下: /** * @describe 使用concat,超出时间限制 ...
- 【图论】CF1508C Complete the MST
Problem Link 有一张 \(n\) 个点的完全图,其中 \(m\) 条边已经标有边权.你需要给剩下的边都标上权值,使得所有边权的异或和为 \(0\),并且整张图的最小生成树边权和最小. \( ...
- 从零开始配置 vim(8)——文件类型检测
在上一章介绍自动命令的时候,我们提到可以使用 FileType来根据文件类型来触发事件,但是关于文件类型并没有深入的介绍,本篇我们来补充关于文件类型相关的内容,让大家更好的理解,看不懂也没关系,你只需 ...
- 5.3 Windows驱动开发:内核取应用层模块基址
在上一篇文章<内核取ntoskrnl模块基地址>中我们通过调用内核API函数获取到了内核进程ntoskrnl.exe的基址,当在某些场景中,我们不仅需要得到内核的基地址,也需要得到特定进程 ...
- JavaScript快速入门(一)
JavaScript快速入门(二) 语句 只需简单地把各条语句放在不同的行上就可以分隔它们 var a = 1 var b = 2 如果想把多条语句放在同一行上,就需要用分号隔开 var a = 1; ...
- 如何使用 etcd 实现分布式 /etc 目录
etcd 是一款兼具一致性和高可用性的键值数据库,简单.安全.快速.可信,目前是 Kubernetes 的首要数据存储.我们先来看一段 etcd 官方对于名字的解释. The name "e ...
- 【题解】P5461 赦免战俘
一.题目 现有 \(2^n\times2^n\ (n≤10)\) 名作弊者站成一个正方形方阵等候 kkksc03 的发落.kkksc03 决定赦免一些作弊者.他将正方形矩阵均分为 4 个更小的正方形矩 ...
- Hadoop集群常用组件的命令
1. Hadoop (1).HDFS:启动HDFS:start-dfs.sh关闭HDFS:stop-dfs.sh格式化NameNode:hdfs namenode -format查看文件系统状态:hd ...
- 下载Apache软件基金的软件和项目(Hadoop相关组件)
一.下载Hadoop相关组件,可以到Apache软件基金的资源目录: Apache 分发目录地址:https://dlcdn.apache.org/ 二.下载软件 方法一:在页面中找到需要下载的软件目 ...
- 2023牛客暑期多校训练营6 ABCEG
比赛链接 A 题解 方法一 知识点:并查集,树形dp,背包dp. 因为需要路径中的最大值,因此考虑按边权从小到大加入图中,保证通过这条边产生贡献的点对已经全部出现. 在加边的同时进行树上背包,答案存在 ...