自定义注解(2)、校验身份证号

在我们日常工作中,会有这样的需求,假如前端新增、修改数据时,后端要针对身份证号格式进行校验 ,但是框架有没有提供现成的关于身份证号校验的注解,如果写在业务层里,无异于会增加代码的复杂度,综上考虑,使用自定义注解是比较方便且高效的

注解代码:

@Documented
@Constraint(validatedBy = IdCardValidator.class)
@Target({ElementType.FIELD})
@Retention(RetentionPolicy.RUNTIME)
public @interface IdCard {

    String message() default "身份证号格式不正确";

    Class<?>[] groups() default {};

    Class<? extends Payload>[] payload() default {};

}

手机号校验业务处理:

public class IdCardValidator implements ConstraintValidator<IdCard, String> {

    private static final Pattern PATTERN = Pattern.compile("^[1-9]\\d{5}(18|19|20)\\d{2}(0[1-9]|1[0-2])(0[1-9]|[1-2][0-9]|3[0-1])\\d{3}([0-9Xx])$");
    
    private static final Map<Integer, String> VERIFY_CODE = new HashMap<Integer, String>() {
        private static final long serialVersionUID = 2725763670639092743L;
        {
            put(0, "1");
            put(1, "0");
            put(2, "X");
            put(3, "9");
            put(4, "8");
            put(5, "7");
            put(6, "6");
            put(7, "5");
            put(8, "4");
            put(9, "3");
            put(10, "2");
        }
    };

    @Override
    public void initialize(IdCard constraintAnnotation) {
        // 初始化方法,通常不需要做额外的处理
    }

    @Override
    public boolean isValid(String value, ConstraintValidatorContext context) {
        if (value == null || value.isEmpty()) {
            return true; // 允许空值通过校验,根据实际需求可以改为false
        }

        Matcher matcher = PATTERN.matcher(value);
        if (!matcher.matches()) {
            return false; // 身份证号格式不正确
        }

        return isValidChineseID(value); // 校验身份证号的校验码
    }

    /**
     * 判断身份证号格式是否正确
     * @param id id
     * @return boolean
     */
    public static boolean isValidChineseID(String id) {
        if (!PATTERN.matcher(id).matches()) {
            return false;
        }

        // 提取身份证的前17位数字
        String idNumber = id.substring(0, 17);
        // 提取身份证的第18位数字或字母
        String lastNumber = id.substring(17, 18);

        // 计算身份证校验码
        int sum = 0;
        for (int i = 0; i < idNumber.length(); i++) {
            sum += Integer.parseInt(String.valueOf(idNumber.charAt(i))) * (Math.pow(2, (idNumber.length() - i)) % 11);
        }

        // 计算出的值对应的校验码
        String verifyCode = VERIFY_CODE.get(sum % 11);

        // 校验码进行比较
        return lastNumber.toUpperCase().equals(verifyCode);
    }

}

实体类使用:

@Data
public class EventRegistrationAddParam {

    @ApiModelProperty(value = "身份证号", example = "身份证号")
    @NotBlank(message = "身份证号不能为空")
    @Length(max = 20, message = "身份证号过长,不能超过20个字符")
    @IdCard
    private String idNumber;

}

控制层:

@PostMapping("add")
@ApiOperation(value = "新增活动报名", notes = "新增活动报名")
public void add(@RequestBody @Validated EventRegistrationAddParam param) {
    charitableOrgBusinessWriteService.addEventRegistration(param, getCurUserAccount());
}
  • 24
    点赞
  • 22
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论
好的,以下是一个自定义注解校验身份证的代码示例: 首先,我们需要定义一个注解 `@IdNumber`,用于标注需要进行身份证校验的字段: ```java @Documented @Constraint(validatedBy = IdNumberValidator.class) @Target({ElementType.FIELD, ElementType.PARAMETER}) @Retention(RetentionPolicy.RUNTIME) public @interface IdNumber { String message() default "身份证号码不正确"; Class<?>[] groups() default {}; Class<? extends Payload>[] payload() default {}; } ``` 其中,`@Constraint(validatedBy = IdNumberValidator.class)` 表示该注解需要使用 `IdNumberValidator` 进行校验。 接着,我们需要实现 `IdNumberValidator`: ```java public class IdNumberValidator implements ConstraintValidator<IdNumber, String> { @Override public void initialize(IdNumber constraintAnnotation) { } @Override public boolean isValid(String value, ConstraintValidatorContext context) { if (StringUtils.isBlank(value)) { return true; } return isIdNumber(value); } /** * 身份证校验 * * @param idNumber 身份证号码 * @return 是否有效 */ private boolean isIdNumber(String idNumber) { String regex = "^\\d{15}$|^\\d{17}[0-9Xx]$"; if (!idNumber.matches(regex)) { return false; } String provinceCode = idNumber.substring(0, 2); if (!ProvinceCodeUtils.isValidProvinceCode(provinceCode)) { return false; } if (!isValidBirthDate(idNumber)) { return false; } return isValidCheckCode(idNumber); } /** * 校验生日日期 * * @param idNumber 身份证号码 * @return 是否有效 */ private boolean isValidBirthDate(String idNumber) { String birthDateStr; if (idNumber.length() == 15) { birthDateStr = "19" + idNumber.substring(6, 12); } else { birthDateStr = idNumber.substring(6, 14); } SimpleDateFormat format = new SimpleDateFormat("yyyyMMdd"); format.setLenient(false); try { format.parse(birthDateStr); } catch (ParseException e) { return false; } return true; } /** * 校验校验码 * * @param idNumber 身份证号码 * @return 是否有效 */ private boolean isValidCheckCode(String idNumber) { if (idNumber.length() == 15) { return true; } char[] idCharArray = idNumber.toCharArray(); int[] idIntArray = new int[17]; for (int i = 0; i < 17; i++) { idIntArray[i] = Integer.parseInt(String.valueOf(idCharArray[i])); } int[] weightFactorArray = new int[]{7, 9, 10, 5, 8, 4, 2, 1, 6, 3, 7, 9, 10, 5, 8, 4, 2}; int sum = 0; for (int i = 0; i < 17; i++) { sum += idIntArray[i] * weightFactorArray[i]; } int checkCodeIndex = sum % 11; char[] checkCodeArray = {'1', '0', 'X', '9', '8', '7', '6', '5', '4', '3', '2'}; char checkCode = checkCodeArray[checkCodeIndex]; return checkCode == idCharArray[17]; } } ``` 其中,`isValid` 方法用于进行身份证校验,`isValidBirthDate` 方法用于校验生日日期,`isValidCheckCode` 方法用于校验校验码。 最后,在需要进行身份证校验的字段上使用 `@IdNumber` 注解即可: ```java public class User { @IdNumber private String idNumber; // getter/setter 略 } ``` 这样,在使用 `@Valid` 注解校验实体时,如果 `idNumber` 字段不符合身份证规则,则会抛出 `ConstraintViolationException` 异常。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

一只鸟儿

你的鼓励是我最大的动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值