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

二代身份证编码规则及校验代码实现本文主要讨论的是二代身份证编码规则及其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)
全栈程序员-站长的头像全栈程序员-站长


相关推荐

  • golang 激活码 2021[在线序列号][通俗易懂]

    golang 激活码 2021[在线序列号],https://javaforall.net/100143.html。详细ieda激活码不妨到全栈程序员必看教程网一起来了解一下吧!

    2022年3月18日
    80
  • poj1195(二维树状数组)

    poj1195(二维树状数组)

    2021年7月6日
    97
  • 闭包概念及面试题

    闭包概念及面试题如何产生闭包(closure)闭包(closure),是指函数变量可以保存在函数作用域内,因此看起来是函数将变量“包裹”了起来。//根据定义,包含变量的函数就是闭包也就是函数嵌套函数就可以称之为闭包.作用域应对的特殊情况,有两种表现:函数作为参数被传递函数作为返回值被带回函数中的自由变量,取决于函数定义的地方,跟执行的地方没关系闭包的应用场景闭包应用场景1,封装对象的私有属性和方法隐藏数据做一个简单的缓存工具//闭包隐藏数据,只提供APIfunctioncreat

    2022年10月9日
    0
  • Linux计划任务「建议收藏」

    Linux计划任务「建议收藏」计划任务的安排方式分两种:一种是定时性的,也就是例行。就是每隔一定的周期就要重复来做这个事情一种是突发性的,就是这次做完了这个事,就没有下一次了,临时决定,只执行一次的任务at和crontab这

    2022年7月4日
    16
  • SQL Server如何备份数据库?完整数据库备份方式

    SQL Server如何备份数据库?完整数据库备份方式第一次写博客嘿嘿,喜欢或是觉得对您有帮助,请帮忙给我点个赞哦!您的认可就是对我最大的支持~蟹蟹

    2022年5月15日
    38
  • java学生成绩管理系统界面设计

    java学生成绩管理系统界面设计关于学生成绩管理系统的界面设计:代码如下1import&nbsp;javax.swing.*;2import&nbsp;java.awt.*;3import&nbsp;java.awt.event.*;4import&nbsp;com.example.SqlHelper;5class&nbsp;MyPanel&nbsp;extends&nbsp;JPanel{6&nbsp;&nbs…

    2022年7月17日
    11

发表回复

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

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