自定义校验注解,优雅的实现手机号,身份证号的格式校验!

导读(皮一下)

为什么需要自定义校验注解?

系统执行业务逻辑之前,会对输入数据进行校验,检测数据是否有效合法的。所以我们可能会写大量的if else等判断逻辑,特别是在不同方法出现相同的数据时,校验的逻辑代码反复出现,导致代码冗余阅读性和可维护性极差

  • 鉴于通用性和普遍性,Spring框架提供了validator组件,通过一些校验器,可以对一些数据进行统一的完整性和有效性等校验,即简单又好用。
  • JSR-303Java为Bean数据合法性校验提供的标准框架,它定义了一整套校验注解,可以标注在成员变量,属性方法等之上。
  • 但是这些通用的校验注解,并不能满足我们的所有需求,如电话号格式校验,身份证号格式校验…

常用的校验器

在Java开发中,校验器(validator)是指用于数据验证的工具或库,它们帮助确保应用程序的数据符合预期的格式或规则。以下是一些常用的校验器库和框架:

名称描述
Bean Validation (JSR 303/349/380)Java平台的标准数据验证API,提供注解驱动的数据验证机制
Hibernate ValidatorBean Validation的参考实现,提供丰富的预定义注解,如@NotNull, @Size等
Spring ValidationSpring框架中的集成Bean Validation的包,提供额外的校验功能
Apache Commons Validator提供用于验证不同数据格式的工具,如电子邮件、URL、IP地址等
Google GuavaGuava库中的Strings类,用于简单的字符串校验
AssertJ一个断言库,用于编写更可读的单元测试,也常用于校验方法参数和对象状态
JSR 303/349/380注解包括@Null, @NotNull, @AssertTrue等,用于标注验证规则
自定义校验器开发者可以自定义校验注解和校验器类,满足特定的业务需求
Spring Boot Starter ValidationSpring Boot提供的验证启动器,集成了Hibernate Validator,简化配置
Lombok自动生成Getter、Setter、构造器等,可配合Bean Validation使用,减少样板代码

自定义校验注解

自定义校验注解通常分为如下几步:

  1. 配置验证框架: 确保项目中包含了Bean Validation API的相关依赖(pom依赖,有没有校验器的jar)
  2. 创建注解,使用@Constraint元注解来声明这是一个约束注解,使用@Target@Retention元注解来指定注解的作用目标和保留策略。
  3. 实现校验器: 创建一个类实现ConstraintValidator接口,这个类将包含实际的校验逻辑。需要重写initialize方法(初始化,基本不咋用到)和isValid方法,isValid方法用于执行具体的校验逻辑。
  4. 使用注解:在需要校验的字段或方法上使用自定义注解。

自定义电话号格式校验注解

import javax.validation.Constraint;
import javax.validation.Payload;
import java.lang.annotation.*;/*** 手机号校验注解*/
@Documented
@Constraint(validatedBy = PhoneNumberValidator.class)
@Target({ElementType.METHOD, ElementType.FIELD})
@Retention(RetentionPolicy.RUNTIME)
public @interface ValidPhoneNumber {String message() default "手机号格式不正确";Class<?>[] groups() default {};Class<? extends Payload>[] payload() default {};
}
// 实现校验逻辑的类
import javax.validation.ConstraintValidator;
import javax.validation.ConstraintValidatorContext;public class PhoneNumberValidator implements ConstraintValidator<ValidPhoneNumber, String> {@Overridepublic void initialize(ValidPhoneNumber constraintAnnotation) {// 初始化方法,可以在这里进行一些初始化工作,但通常不需要实现}@Overridepublic boolean isValid(String phoneNumber, ConstraintValidatorContext context) {// 正则表达式,用于校验手机号格式String regex = "^1[3-9]\\d{9}$";return phoneNumber.matches(regex);}
}

使用方式:

public class User {@ValidPhoneNumber(message = "请输入正确的手机号")private String phone;// ...
}

自定义身份证号格式校验注解

import javax.validation.Constraint;
import javax.validation.Payload;
import java.lang.annotation.*;@Documented
@Constraint(validatedBy = IDCardValidator.class)
@Target({ElementType.METHOD, ElementType.FIELD})
@Retention(RetentionPolicy.RUNTIME)
public @interface ValidIDCard {String message() default "身份证号码格式不正确";Class<?>[] groups() default {};Class<? extends Payload>[] payload() default {};
}
import javax.validation.ConstraintValidator;
import javax.validation.ConstraintValidatorContext;public class IDCardValidator implements ConstraintValidator<ValidIDCard, String> {private static final int ID_CARD_LENGTH_18 = 18;private static final String ID_CARD_REGEX_18 = "^[1-9]\\d{5}(18|19|20)\\d{2}(0[1-9]|1[012])(0[1-9]|[12][0-9]|3[01])\\d{3}([0-9]|X)$";private static final char[] VERIFY_CODE = {'1', '0', 'X', '9', '8', '7', '6', '5', '4', '3', '2'};private static final int[] WEIGHTS = {7, 9, 10, 5, 8, 4, 2, 1, 6, 3, 7, 9, 10, 5, 8, 4, 2};@Overridepublic void initialize(ValidIDCard constraintAnnotation) {}@Overridepublic boolean isValid(String idCard, ConstraintValidatorContext context) {if (idCard == null || idCard.length() != ID_CARD_LENGTH_18) {return false;}if (!idCard.matches(ID_CARD_REGEX_18)) {return false;}// 验证18位身份证的校验码return validate18(idCard);}private boolean validate18(String idCard) {int sum = 0;for (int i = 0; i < 17; i++) {sum += Character.getNumericValue(idCard.charAt(i)) * WEIGHTS[i];}int mod = sum % 11;char checkCode = VERIFY_CODE[mod];return Character.toUpperCase(idCard.charAt(17)) == checkCode;}
}

使用方式:

import javax.validation.constraints.NotNull;
import javax.validation.Valid;public class Person {@NotNull(message = "姓名不能为空")private String name;@ValidIDCard(message = "身份证号码格式不正确")private String idCard;// ...
}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/web/50178.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

数据结构 -- 算法的时间复杂度和空间复杂度

数据结构 -- 算法的时间复杂度和空间复杂度 1.算法效率1.1 如何衡量一个算法的好坏1.2 算法的复杂度 2.时间复杂度2.1 时间复杂度的概念2.2 大O的渐进表示法2.3常见时间复杂度计算举例 3.空间复杂度4. 常见复杂度对比 1.算法效率 1.1 如何衡量一个算法的好坏 如何衡量一个算法…

我在高职教STM32——EXTI之外部按键中断(1)

大家好,我是老耿,高职青椒一枚,一直从事单片机、嵌入式、物联网等课程的教学。对于高职的学生层次,同行应该都懂的,老师在课堂上教学几乎是没什么成就感的。正是如此,才有了借助CSDN平台寻求认同感和成就感的想法。在这里,我准备陆续把自己花了很多心思设计的教学课件分…

人工智能学习①

LLM背景知识介绍 大语言模型 (LLM) 背景 用于理解和生成人类语言&#xff0c;能够处理诸如文本分类、问答、翻译和对话等多种自然语言任务。 语言模型 (Language Model, LM) &#xff1a;给定一个短语&#xff08;一个词组或者一句话&#xff09;语言模型可以生成&#xff0…

Windows下Git Bash乱码问题解决

Windows下Git Bash乱码问题解决 缘起 个人用的电脑是Mac OS&#xff0c;系统和终端编码都是UTF-8&#xff0c;但公司给配发的电脑是Windows&#xff0c;装上Git Bash在使用 git commit -m "中文"时会乱码 解决 确认有以下配置 # 输入 git config --global --lis…

网络安全的重要性与挑战

网络安全的重要性与挑战 网络安全是指保护网络系统中的数据免受未经授权的访问、泄露、篡改或破坏的过程。它关乎个人隐私的保护、企业资产的安全乃至国家安全的稳定。随着互联网和物联网的快速发展&#xff0c;网络攻击和威胁也越来越多样化和复杂化&#xff0c;网络安全已成为…

【第六天】TCP和UDP的区别 TCP连接如何确保可靠性

TCP和UDP的区别 概念&#xff1a; TCP&#xff08;传输控制协议&#xff09;是一种面向连接的、可靠的、基于字节流的传输层通信协议UDP&#xff08;用户数据报协议&#xff09;为应用程序提供了一种无需建立连接就可以发送封装的IP数据包的方法。 特点&#xff1a; TCP&am…

深入Scrapy框架:掌握其工作流程

深入Scrapy框架&#xff1a;掌握其工作流程 引言 作为一名资深的Python程序员&#xff0c;我对各种数据采集工具有着深刻的理解。Scrapy&#xff0c;作为一个上场率极高的爬虫框架&#xff0c;以其高效、灵活和强大的特性&#xff0c;成为数据采集领域的不二选择。在本文中&a…

CSS常用属性(列表属性、表格属性、背景属性、鼠标属性)

一、CSS列表属性 列表相关的属性&#xff0c;可以作用在 ul 、 ol 、 li 元素上。 CSS 属性名 功能 属性值 list - style - type 设置列表符号 常用值如下&#xff1a; none &#xff1a;不显示前面的标识&#xff08;很常用&#xff01;&#xff09; square &#xf…

Unity 物理动画:利用物理引擎创造逼真动作

在Unity中&#xff0c;物理动画是一种利用物理引擎来模拟真实世界物理效果的动画技术。通过物理动画&#xff0c;开发者可以创造出更加逼真和自然的动画效果&#xff0c;如重力、碰撞、布料摆动等。本文将介绍Unity物理动画的基本概念、实现方法以及一些实用的技巧。 Unity物理…

Oat++ 后端实现跨域

这里记录在官方的例子中&#xff0c;加入跨域。Oat Example-CRUD 在官方的例子中&#xff0c;加入跨域。 Oat Example-CRUD 修改AppComponent.hpp文件中的代码&#xff0c;如下&#xff1a; #include "AppComponent.hpp"#include "controller/UserController…

路径规划——Dijkstra算法

算法原理 Dijkstra算法采用贪心算法的思想&#xff0c;解决的问题可以描述为&#xff1a;在无向图G(V,E)中&#xff0c;假设每条边E[i] 的长度为 w[i]&#xff0c;找到由顶点vs到其余各点的最短路径。 通过Dijkstra计算图G中的最短路径时&#xff0c;需要指定起点vs(即从顶点v…

BI数据可视化看板的力量与应用

在当今数据驱动的时代&#xff0c;企业面对着海量的信息与数据。随着业务的复杂性加剧&#xff0c;如何有效地解读和利用这些数据&#xff0c;成为了企业决策的重要环节。而BI&#xff08;商业智能&#xff09;数据可视化看板&#xff0c;便是解决这一难题的关键工具。数聚将深…

idea-springboot后端所有@注释含义汇总-持续更新!

&#xff08;1&#xff09;启动类 ①SpringBootApplication 出现这个代表这个就是整个程序的入口&#xff0c;是运行的开始位置 &#xff08;2&#xff09;Dao层 ①Repository 作用就是声明自己这个为bean文件&#xff08;每一个controller都是一个bean文件&#xff09;&am…

vue3 中使用xlsx 插件 导出excel文件

介绍一下 vue中得导出excel 文件 功能 ① 安装插件 npm i xlsx ②导入插件 import * as XLSX from xlsx; ③ 使用插件 直接是一个 方法 const exportExcel()>{const data[["姓名","年龄"],["张三",18],["李四",20],["王五…

blender顶点乱飞的问题解决

初学blender&#xff0c;编辑模式下移动某些顶点&#xff0c;不管是移动还是滑动都会出现定点乱飞的问题&#xff0c;后来才发现是开了吸附工具的原因&#xff01;&#xff01;&#xff01;&#xff01; 像下面这样&#xff0c;其实我只是在Z轴上移动&#xff0c;但是就跑的很…

Anaconda目录

安装目录 Anaconda 在默认情况下会安装到 C:\ProgramData\Anaconda3&#xff0c;而 conda 环境和包会安装在 C:\Users\username\.conda\ 目录下。 备注&#xff1a;我是在windows下安装 的Anaconda。我的安装目录是C:\Program Files\Anaconda3 pkgs目录 在以上两个目录下都有…

CH571F基于官方模版创建工程

直接使用MounRiver创建的工程只有最简单的串口和GPIO功能&#xff0c;其他PWM和SPI等驱动基本上都有&#xff0c;但蓝牙和USB只有参考官方的示例来&#xff0c;全部自己写属实有点麻烦了&#xff0c;而且还需要添加BLE的库。下面就简单基于官方的示例工程创建我们自己的工程。 …

抓包工具——wireshark的使用

​ 什么是wireshark wireshark是一个数据包捕捉程序。和linux下的tcpdump&#xff0c;以及sniffer&#xff0c;Fidder等软件功能类似。按理说&#xff0c;我们的计算机中的网卡设备只会将发给本机的数据包传输到上层进行解析&#xff0c;而其他的数据包会进行丢弃&#xff0c;…

jenkins集成allure测试报告

1.allure插件安装 &#xff08;1&#xff09;点击首页的【Manage Jenkins】-【Manage Plugins】 &#xff08;2&#xff09;选择【Available】选项&#xff0c;搜索输入框输入Allure&#xff0c;搜索出来的名字就叫Allure&#xff0c;当安装后名字会变为Allure Jenkins Plugi…

QVariantMap是QVariant类型的键值对容器,它在 Qt 中被广泛使用,用于存储和传递复杂的数据结构

QVariantMap 是 QVariant 类型的键值对容器&#xff0c;它在 Qt 中被广泛使用&#xff0c;用于存储和传递复杂的数据结构。QVariantMap 类似于 QMap<QString, QVariant>&#xff0c;允许使用字符串作为键来存储各种类型的数据。 QVariantMap 的基本功能 创建和使用 QVa…