二代身份证编码规则及校验代码实现

二代身份证编码规则及校验代码实现本文主要讨论的是二代身份证编码规则及其Java代码实现,下面的校验方式还不是特别严谨,由于只校验了前两位的省份信息,中间六位的出生日期信息和最后一位的校验码信息,故对于部分不满足要求的证件号码刚好同时满足了这里提到的几个条件,也会被判定为是合法的证件号码…1二代身份证号码编码规则1.1编码格式1999年我国颁发了第二代居民身份证号,公民身份号码为18位,且终身不变。居民身份证格式如:ABCDEF

大家好,又见面了,我是你们的朋友全栈君。

本文主要讨论的是二代身份证编码规则及其Java代码实现,下面的校验方式还不是特别严谨,由于只校验了前两位的省份信息,中间六位的出生日期信息和最后一位的校验码信息,故对于部分不满足要求的证件号码刚好同时满足了这里提到的几个条件,也会被判定为是合法的证件号码…

1 二代身份证号码编码规则

1.1 编码格式

1999年我国颁发了第二代居民身份证号,公民身份号码为18位,且终身不变。

居民身份证格式如:ABCDEFYYYYMMDDXXXR

1.1.1地址码(ABCDEF)

表示登记户口时所在地的行政区划代码(省、市、县),如果行政区划进行了重新划分,同一个地方进行户口登记的可能存在地址码不一致的情况。行政区划代码按GB/T2260的规定执行。

1.1.2 出生日期码(YYYYMMDD)

表示该居民的出生年月日,年4位数字,月和日分别用2位数字表示,如19491001,;出生日期码是按GB/T 7408的规定执行的。

1.1.3 顺序码(XXX)

表示同一地址码区域内,同年、同月、同日生的人所编订的顺序号,根据自己身份证的顺序码就可以知道:与我们同年同月同日生的同性至少有多少个,且在我们之前登记户籍的有多少人。身份证顺序码的奇数分配给男性,偶数分配给女性。这就是为什么倒数第二位奇数表示男生,偶数表示女生。

1.1.4 校验码(R)

R之前的17位被称为本体码,R是根据本体码,按照校验码算法(ISO 7064:1983,MOD 11-2校)计算出来的。当我们输入身份号码进行实名认证的时候,根据校验码算法可以初步判断你输入身份证号码格式是否正确。

1.2 校验码算法

将本体码各位数字乘以对应加权因子并求和,除以11得到余数,根据余数通过校验码对照表查得校验码。

1.2.1 加权因子

位置序号 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17
加权因子 7 9 10 5 8 4 2 1 6 3 7 9 10 5 8 4 2

(本体码每个位置对应的加权因子)

1.2.2 校验码表

余数 0 1 2 3 4 5 6 7 8 9 10
校验码 1 0 X 9 8 7 6 5 4 3 2

(每个余数对应的校验码)

1.2.3 应用举例

某公民的身份证号码是34052419800101001X

第一步:本体码乘以加权因子:

3*7+4*9+0*10+……0*4+1*2=189

第二步:计算求和后除以11的余数

189%11=2

第三步:在检验码中查询余数对应的检验码

2所对应的校验码是X,注意X必须大写

2 Java编码实现

2.1 对外提供的调用接口

/** * 二代身份证号码有效性校验 * * @param idNo * @return */
public static boolean isValidIdNo(String idNo) {
  return isIdNoPattern(idNo) && isValidProvinceId(idNo.substring(0, 2))
      && isValidDate(idNo.substring(6, 14)) && checkIdNoLastNum(idNo);
}

2.2 二代身份证正则表达式

/** * 二代身份证正则表达式 * * @param idNo * @return */
private static boolean isIdNoPattern(String idNo) {
  return Pattern.matches("^[1-9]\\d{5}[1-9]\\d{3}((0\\d)|(1[0-2]))(([0|1|2]\\d)|3[0-1])\\d{3}([\\d|x|X]{1})$", idNo);
}

2.3 校验前两位省份信息

2.3.1 常量

//省(直辖市)码表
private static String provinceCode[] = { "11", "12", "13", "14", "15", "21", "22",
        "23", "31", "32", "33", "34", "35", "36", "37", "41", "42", "43",
        "44", "45", "46", "50", "51", "52", "53", "54", "61", "62", "63",
        "64", "65", "71", "81", "82", "91" };

2.3.2 检查省份信息

/** * 检查身份证的省份信息是否正确 * @param provinceId * @return */
public static boolean isValidProvinceId(String provinceId){
    for (String id : provinceCode) {
        if (id.equals(provinceId)) {
            return true;
        }
    }
    return false;
}

2.4 判断中间的六位日期是否有效

/** * 判断日期是否有效 * @param inDate * @return */
 public static boolean isValidDate(String inDate) {
    if (inDate == null){
      return false;
    }
    SimpleDateFormat dateFormat = new SimpleDateFormat("yyyyMMdd");
    if (inDate.trim().length() != dateFormat.toPattern().length()){
          return false;
    }
    dateFormat.setLenient(false);//执行严格的日期匹配
    try {
        dateFormat.parse(inDate.trim());
    } catch (ParseException e) {
        return false;
    }
    return true;
}

2.4 校验第18位校验码

2.4.1 常量

//身份证前17位每位加权因子
private static int[] power = {
  
  7, 9, 10, 5, 8, 4, 2, 1, 6, 3, 7, 9, 10, 5, 8, 4, 2};

//身份证第18位校检码
private static String[] refNumber ={
  
  "1", "0", "X", "9", "8", "7", "6", "5", "4", "3", "2"};

2.4.2 计算第18位校验码

/** * 计算身份证的第十八位校验码 * @param cardIdArray * @return */
public static String sumPower(int[] cardIdArray){
    int result = 0;
    for(int i=0;i<power.length;i++){
        result += power[i] * cardIdArray[i];
    }
    return refNumber[(result%11)];
}

2.4.2 验证第18位校验码是否正确

/** * 校验身份证第18位是否正确(只适合18位身份证) * @param idNo * @return */
public static boolean checkIdNoLastNum(String idNo){
    if(idNo.length() != 18){
        return false;
    }
    char[] tmp = idNo.toCharArray();
    int[] cardidArray = new int[tmp.length-1];
    int i=0;
    for(i=0;i<tmp.length-1;i++){
        cardidArray[i] = Integer.parseInt(tmp[i]+"");
    }
    String checkCode = sumPower(cardidArray);
    String lastNum = tmp[tmp.length-1] + "";
    if(lastNum.equals("x")){
        lastNum = lastNum.toUpperCase();
    }
    if(!checkCode.equals(lastNum)){
        return false;
    }
    return true;
}

3 参考文献

金融支付:身份证号码的编码规则
java对身份证验证及正则表达式解析

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请联系我们举报,一经查实,本站将立刻删除。

发布者:全栈程序员-站长,转载请注明出处:https://javaforall.net/153655.html原文链接:https://javaforall.net

(0)
全栈程序员-站长的头像全栈程序员-站长


相关推荐

  • linux小红帽网卡设定,linux下网卡如何开启flow control「建议收藏」

    linux小红帽网卡设定,linux下网卡如何开启flow control「建议收藏」Whatisawife?Beyouwouldliketohandovertoherthesavingsthepreservationofwoman.Whatisalover?Beyougosecretivelyandshedatesagainafraidthewiferunintoofwoman.Whatisabeauty…

    2022年5月18日
    44
  • java算法是什么_什么是java算法

    java算法是什么_什么是java算法什么是java算法算法是指解题方案的准确而完整的描述,是一系列解决问题的清晰指令,java算法就是采用Java语言来实现解决某一问题的清晰指令。算法的特征:输入性:有零个或多个外部量作为算法的输入输出性:算法产生至少一个量作为输出确定性:算法中每条指令清晰,无歧义有穷性:算法中每条指令的执行次数有限,执行每条指令是时间也有限可行性:算法原则上能够精确的运行,而且人们用纸和笔做有限次运算后即可完成程…

    2022年7月9日
    23
  • Go如何发送广播包

    Go如何发送广播包

    2022年1月6日
    54
  • 腾讯股票数据接口

     0:未知 1:名字:name 2:代码:code 3:现价:trade 4:昨收:yestclose 5:今开:open 6:成交量(手):volume 7:外盘 8:内盘 9:买一10:买一量(手)11-18:买二买五19:卖一20:卖一量21-28:卖二卖五29:最近逐笔成交30:时间31:涨跌:change32:涨跌幅:changepercent33:最高:high34:最低:low3…

    2022年4月8日
    183
  • pnp饱和状态条件_二极管的极性判别

    pnp饱和状态条件_二极管的极性判别1.截止状态:当b-e结反偏、零偏、浅正偏(指琐然正偏,似正向压降小于门槛电压)时Ib=0.Ic=Ice≈0,三极管截止,此时Rce内阻为无穷大,Vce约等于电源电压。  2.导通放大状态:当be正偏、bc反偏;且Vbe大于门槛电压时为导通放大状态,此时Vc>Vb>Ve.且Vbe=0.7V。导通后Rce减小,导通越深,Rce内阻越小.Vce越小。在放大状态,Ib能控制Ic.即具有电…

    2025年9月7日
    5
  • IIS与Apache同时使用80端口

    IIS与Apache同时使用80端口

    2021年10月8日
    41

发表回复

您的邮箱地址不会被公开。 必填项已用 * 标注

关注全栈程序员社区公众号