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

简介: 本文介绍了如何创建自定义校验注解来实现手机号和身份证号的格式校验,包括定义注解、实现校验逻辑、将注解应用于模型类,以及常用的校验器库和框架,旨在提高代码的可维护性和减少重复的校验逻辑。

导读(皮一下)

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

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

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

常用的校验器

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

名称 描述
Bean Validation (JSR 303/349/380) Java平台的标准数据验证API,提供注解驱动的数据验证机制
Hibernate Validator Bean Validation的参考实现,提供丰富的预定义注解,如@NotNull, @Size等
Spring Validation Spring框架中的集成Bean Validation的包,提供额外的校验功能
Apache Commons Validator 提供用于验证不同数据格式的工具,如电子邮件、URL、IP地址等
Google Guava Guava库中的Strings类,用于简单的字符串校验
AssertJ 一个断言库,用于编写更可读的单元测试,也常用于校验方法参数和对象状态
JSR 303/349/380注解 包括@Null, @NotNull, @AssertTrue等,用于标注验证规则
自定义校验器 开发者可以自定义校验注解和校验器类,满足特定的业务需求
Spring Boot Starter Validation Spring 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> {
   
    @Override
    public void initialize(ValidPhoneNumber constraintAnnotation) {
   
        // 初始化方法,可以在这里进行一些初始化工作,但通常不需要实现
    }

    @Override
    public 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};

    @Override
    public void initialize(ValidIDCard constraintAnnotation) {
   }

    @Override
    public 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;

    // ...
}

相关文章
|
10天前
|
弹性计算 人工智能 架构师
阿里云携手Altair共拓云上工业仿真新机遇
2024年9月12日,「2024 Altair 技术大会杭州站」成功召开,阿里云弹性计算产品运营与生态负责人何川,与Altair中国技术总监赵阳在会上联合发布了最新的“云上CAE一体机”。
阿里云携手Altair共拓云上工业仿真新机遇
|
7天前
|
机器学习/深度学习 算法 大数据
【BetterBench博士】2024 “华为杯”第二十一届中国研究生数学建模竞赛 选题分析
2024“华为杯”数学建模竞赛,对ABCDEF每个题进行详细的分析,涵盖风电场功率优化、WLAN网络吞吐量、磁性元件损耗建模、地理环境问题、高速公路应急车道启用和X射线脉冲星建模等多领域问题,解析了问题类型、专业和技能的需要。
2513 16
【BetterBench博士】2024 “华为杯”第二十一届中国研究生数学建模竞赛 选题分析
|
7天前
|
机器学习/深度学习 算法 数据可视化
【BetterBench博士】2024年中国研究生数学建模竞赛 C题:数据驱动下磁性元件的磁芯损耗建模 问题分析、数学模型、python 代码
2024年中国研究生数学建模竞赛C题聚焦磁性元件磁芯损耗建模。题目背景介绍了电能变换技术的发展与应用,强调磁性元件在功率变换器中的重要性。磁芯损耗受多种因素影响,现有模型难以精确预测。题目要求通过数据分析建立高精度磁芯损耗模型。具体任务包括励磁波形分类、修正斯坦麦茨方程、分析影响因素、构建预测模型及优化设计条件。涉及数据预处理、特征提取、机器学习及优化算法等技术。适合电气、材料、计算机等多个专业学生参与。
1520 14
【BetterBench博士】2024年中国研究生数学建模竞赛 C题:数据驱动下磁性元件的磁芯损耗建模 问题分析、数学模型、python 代码
|
3天前
|
存储 关系型数据库 分布式数据库
GraphRAG:基于PolarDB+通义千问+LangChain的知识图谱+大模型最佳实践
本文介绍了如何使用PolarDB、通义千问和LangChain搭建GraphRAG系统,结合知识图谱和向量检索提升问答质量。通过实例展示了单独使用向量检索和图检索的局限性,并通过图+向量联合搜索增强了问答准确性。PolarDB支持AGE图引擎和pgvector插件,实现图数据和向量数据的统一存储与检索,提升了RAG系统的性能和效果。
|
9天前
|
编解码 JSON 自然语言处理
通义千问重磅开源Qwen2.5,性能超越Llama
击败Meta,阿里Qwen2.5再登全球开源大模型王座
545 14
|
1月前
|
运维 Cloud Native Devops
一线实战:运维人少,我们从 0 到 1 实践 DevOps 和云原生
上海经证科技有限公司为有效推进软件项目管理和开发工作,选择了阿里云云效作为 DevOps 解决方案。通过云效,实现了从 0 开始,到现在近百个微服务、数百条流水线与应用交付的全面覆盖,有效支撑了敏捷开发流程。
19282 30
|
9天前
|
人工智能 自动驾驶 机器人
吴泳铭:AI最大的想象力不在手机屏幕,而是改变物理世界
过去22个月,AI发展速度超过任何历史时期,但我们依然还处于AGI变革的早期。生成式AI最大的想象力,绝不是在手机屏幕上做一两个新的超级app,而是接管数字世界,改变物理世界。
464 48
吴泳铭:AI最大的想象力不在手机屏幕,而是改变物理世界
|
1月前
|
人工智能 自然语言处理 搜索推荐
阿里云Elasticsearch AI搜索实践
本文介绍了阿里云 Elasticsearch 在AI 搜索方面的技术实践与探索。
18838 20
|
1月前
|
Rust Apache 对象存储
Apache Paimon V0.9最新进展
Apache Paimon V0.9 版本即将发布,此版本带来了多项新特性并解决了关键挑战。Paimon自2022年从Flink社区诞生以来迅速成长,已成为Apache顶级项目,并广泛应用于阿里集团内外的多家企业。
17527 13
Apache Paimon V0.9最新进展
|
1天前
|
云安全 存储 运维
叮咚!您有一份六大必做安全操作清单,请查收
云安全态势管理(CSPM)开启免费试用
360 4
叮咚!您有一份六大必做安全操作清单,请查收