身份证正则判断最准确的判断方式,没有之一!

/**
 * Create by Mazhanzhu on 2020/9/28
 * 身份证号码校验
 */
public class IDCardValidate {
    public static final String[] ValCodeArr = {"1", "0", "x", "9", "8", "7", "6", "5", "4", "3", "2"};
    public static final String[] Wi = {"7", "9", "10", "5", "8", "4", "2", "1", "6", "3", "7", "9", "10", "5", "8", "4", "2"};
    // 身份证的最小出生日期,1900年1月1日
    private final static Date MINIMAL_BIRTH_DATE = new Date(-2209017600000L);
    private static final String BIRTH_DATE_FORMAT = "yyyyMMdd";
    private final static int NEW_CARD_NUMBER_LENGTH = 18;
    private final static int OLD_CARD_NUMBER_LENGTH = 15;
    private final static String LENGTH_ERROR = "身份证长度必须为15或者18位!";
    private final static String NUMBER_ERROR = "15位身份证都应该为数字,18位身份证都应该前17位应该都为数字!";
    private final static String DATE_ERROR = "身份证日期验证无效!";
    private final static String AREA_ERROR = "身份证地区编码错误!";
    private final static String CHECKCODE_ERROR = "身份证最后一位校验码有误!";
    //是否需要返回自动补全成的身份证
    private static boolean isNeedReturn_AutoCard = false;

    /**
     * @param idcardNumber      需要验证的身份证
     * @param isreturn_AutoCard 验证无误后,是否需要返回自动补全身份证
     * @return 身份证无误返回传入的身份证号
     */
    public static boolean validate_effective(String idcardNumber, boolean isreturn_AutoCard) {
        isNeedReturn_AutoCard = isreturn_AutoCard;
        return validate_effective(idcardNumber);
    }

    /**
     * 身份证校验
     *
     * @param idcardNumber 需要验证的身份证
     * @return 身份证无误返回传入的身份证号
     */
    public static boolean validate_effective(String idcardNumber) {
        String Ai = idcardNumber.trim();
        System.out.println(Ai.length() != 15);
        if (Ai.length() == 15 | Ai.length() == 18) {
            //如果为15位则自动补全到18位
            if (Ai.length() == OLD_CARD_NUMBER_LENGTH) {
                Ai = contertToNewCardNumber(Ai);
            }
        } else {
            return false;
        }
        // 身份证号的前15,17位必须是阿拉伯数字
        for (int i = 0; i < NEW_CARD_NUMBER_LENGTH - 1; i++) {
            char ch = Ai.charAt(i);
            if (ch < '0' || ch > '9') {
                return false;
            }
        }
        //校验身份证日期信息是否有效 ,出生日期不能晚于当前时间,并且不能早于1900年
        try {
            Date birthDate = getBirthDate(Ai);
            if (null == birthDate) {
                return false;
            }
            if (!birthDate.before(new Date())) {
                return false;
            }
            if (!birthDate.after(MINIMAL_BIRTH_DATE)) {
                return false;
            }
            /**
             * 出生日期中的年、月、日必须正确,比如月份范围是[1,12],日期范围是[1,31],还需要校验闰年、大月、小月的情况时,
             * 月份和日期相符合
             */
            String birthdayPart = getBirthDayPart(Ai);
            String realBirthdayPart = createBirthDateParser().format(birthDate);
            if (!birthdayPart.equals(realBirthdayPart)) {
                return false;
            }
        } catch (Exception e) {
            return false;
        }
        //校验地区码是否正确
        Hashtable<String, String> h = GetAreaCode();
        if (h.get(Ai.substring(0, 2)) == null) {
            return false;
        }
        //校验身份证最后一位 身份证校验码
        if (!calculateVerifyCode(Ai).equals(String.valueOf(Ai.charAt(NEW_CARD_NUMBER_LENGTH - 1)))) {
            return false;
        }
        return true;
    }

    /**
     * 把15位身份证号码转换到18位身份证号码<br>
     * 15位身份证号码与18位身份证号码的区别为:<br>
     * 1、15位身份证号码中,"出生年份"字段是2位,转换时需要补入"19",表示20世纪<br>
     * 2、15位身份证无最后一位校验码。18位身份证中,校验码根据根据前17位生成
     *
     * @param oldCardNumber
     * @return
     */
    private static String contertToNewCardNumber(String oldCardNumber) {
        StringBuilder buf = new StringBuilder(NEW_CARD_NUMBER_LENGTH);
        buf.append(oldCardNumber.substring(0, 6));
        buf.append("19");
        buf.append(oldCardNumber.substring(6));
        buf.append(calculateVerifyCode(buf));
        return buf.toString();
    }

    /**
     * 计算最后一位校验码  加权值%11
     * (1)十七位数字本体码加权求和公式 S = Sum(Ai * Wi), i = 0, ... , 16 ,先对前17位数字的权求和
     * Ai:表示第i位置上的身份证号码数字值 Wi:表示第i位置上的加权因子 Wi: 7 9 10 5 8 4 2 1 6 3 7 9 10 5 8 4
     * (2)计算模 Y = mod(S, 11)
     * (3)通过模得到对应的校验码 Y: 0 1 2 3 4 5 6 7 8 9 10 校验码: 1 0 X 9 8 7 6 5 4 3 2
     *
     * @param cardNumber
     * @return
     */
    private static String calculateVerifyCode(CharSequence cardNumber) {
        int sum = 0;
        for (int i = 0; i < NEW_CARD_NUMBER_LENGTH - 1; i++) {
            char ch = cardNumber.charAt(i);
            sum += ((int) (ch - '0')) * Integer.parseInt(Wi[i]);
        }
        return ValCodeArr[sum % 11];
    }

    /**
     * 功能:设置地区编码
     *
     * @return Hashtable 对象
     */
    private static Hashtable<String, String> GetAreaCode() {
        Hashtable<String, String> hashtable = new Hashtable<String, String>();
        hashtable.put("11", "北京");
        hashtable.put("12", "天津");
        hashtable.put("13", "河北");
        hashtable.put("14", "山西");
        hashtable.put("15", "内蒙古");
        hashtable.put("21", "辽宁");
        hashtable.put("22", "吉林");
        hashtable.put("23", "黑龙江");
        hashtable.put("31", "上海");
        hashtable.put("32", "江苏");
        hashtable.put("33", "浙江");
        hashtable.put("34", "安徽");
        hashtable.put("35", "福建");
        hashtable.put("36", "江西");
        hashtable.put("37", "山东");
        hashtable.put("41", "河南");
        hashtable.put("42", "湖北");
        hashtable.put("43", "湖南");
        hashtable.put("44", "广东");
        hashtable.put("45", "广西");
        hashtable.put("46", "海南");
        hashtable.put("50", "重庆");
        hashtable.put("51", "四川");
        hashtable.put("52", "贵州");
        hashtable.put("53", "云南");
        hashtable.put("54", "西藏");
        hashtable.put("61", "陕西");
        hashtable.put("62", "甘肃");
        hashtable.put("63", "青海");
        hashtable.put("64", "宁夏");
        hashtable.put("65", "新疆");
        hashtable.put("71", "台湾");
        hashtable.put("81", "香港");
        hashtable.put("82", "澳门");
        hashtable.put("91", "国外");
        return hashtable;
    }

    private static Date getBirthDate(String idcard) {
        Date cacheBirthDate = null;
        try {
            cacheBirthDate = createBirthDateParser().parse(getBirthDayPart(idcard));
        } catch (Exception e) {
            throw new RuntimeException("身份证的出生日期无效");
        }
        return new Date(cacheBirthDate.getTime());
    }

    private static SimpleDateFormat createBirthDateParser() {
        return new SimpleDateFormat(BIRTH_DATE_FORMAT);
    }

    private static String getBirthDayPart(String idcardnumber) {
        return idcardnumber.substring(6, 14);
    }
}

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 2
    评论
使用正则表达式判断身份证号码的方法如下: 1. 首先,定义一个正则表达式,该正则表达式可以匹配符合身份证号码格式的字符串。根据提供的引用中的代码,我们可以看到正则表达式的定义为: /^[1-9]\d{5}(19|20)\d{2}((0[1-9]|1[0-2]))(([0-2][1-9])|10|20|30|31)\d{3}[0-9Xx]$/ 这个正则表达式的具体含义是: - ^ 表示匹配字符串的开始位置 - [1-9] 表示第一位是非零数字 - \d{5} 表示接下来的五位数字可以是任意数字 - (19|20) 表示年份的前两位可以是19或者20 - \d{2} 表示接下来的两位数字可以是任意数字 - ((0[1-9]|1[0-2])) 表示月份可以是01到12 - (([0-2][1-9])|10|20|30|31) 表示日期可以是01到31 - \d{3} 表示接下来的三位数字可以是任意数字 - [0-9Xx] 表示最后一位可以是数字0到9,或者字母X(大小写不敏感) - $ 表示匹配字符串的结束位置 所以,这个正则表达式可以用来判断一个字符串是否是符合身份证号码格式的。 2. 其次,将定义好的正则表达式应用到需要判断身份证号码上。根据提供的引用中的代码,我们可以看到在输入框失去焦点时,会调用一个函数来判断输入框中的值是否符合身份证号码格式。具体判断的代码如下: if (reg.test(card.value)) { // 符合身份证号格式 // ... } else { // 不符合身份证号格式 // ... } 这段代码首先使用正则表达式的test方法对输入框的值进行匹配,如果匹配成功即返回true,表示输入的字符串符合身份证号码格式;如果匹配失败即返回false,表示输入的字符串不符合身份证号码格式。 3. 最后,根据匹配结果进行相应的操作。根据提供的引用中的代码,我们可以看到如果输入的字符串符合身份证号码格式,会给一个span元素添加一个ok类,并显示"身份证号格式正确~"的提示信息;如果输入的字符串不符合身份证号码格式,会给span元素添加一个fail类,并显示"身份证号格式错误~"的提示信息。 所以,以上就是使用正则表达式判断身份证号码的方法。

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值