身份证的编码规则

原文:https://www.fanhaobai.com/2017/08/id-card.html

在我国现行的身份证系统中共有 15 位和 18 位两种身份证号码,第一代身份证大多为 15 位号码,由于 15 位身份证只能为 1900.01.01 到 1999.12.31 出生的人编码(千年虫问题),所以后来逐步替换为 18 位的身份证号码。

编码规则

15位

15 位身份证编码规则为:DDDDDD YYMMDD XXS

各组成部分说明:

部分名 描述
DDDDDD 6 位地区编码
YYMMDD 出生年月。年份用 2 位表示
XXS 顺序码。其中 S 为性别识别码,奇数为男,偶数为女

例如某个 15 位 ID 为:513701930509101。

18位

18 位身份证较 15 位身份证,出生年月改变为 8 位,并引入了校验位。编码规则为:DDDDDD YYYYMMDD XXX Y

各组成部分说明:

部分名 描述
DDDDDD 6 位地区编码
YYYYMMDD 出生年月。年份用 4 位表示
XXX 顺序码。奇数为男,偶数为女
Y 校验位。前 17 位值计算而得

校验位 Y 取值范围为 [1, 0, X, 9, 8, 7, 6, 5, 4, 3, 2],其采用加权方式校验,校验规则为:p = mod(∑(Ai×Wi), 11)

参数说明:

  • i 为身份证数字所在的位数,1-17;
  • Ai 为身份证第 i 位对应的数字值;
  • Wi 为加权因子,值为 [7, 9, 10, 5, 8, 4, 2, 1, 6, 3, 7, 9, 10, 5, 8, 4, 2];
  • p 表示获取 Y 范围值的第 p+1 个值作为校验值 Y;

例如某个 18 位 ID 位:513701199305091010,校验位后续计算得出。

格式校验

通过分析身份证的 编码规则,我们就可以得出身份证的校验规则,这里使用正则表达式去进行匹配。

15位

15 位身份证DDDDDD YYMMDD XXS的每部分的正则匹配表达式为:

部分名 正则表达式
DDDDDD [1-9]\d{5}
YYMMDD (\d{2})(0[1-9]|(1[0-2]))(([0-2][1-9])|([1-2]0)|31)
XXS \d{3}

由此可得 15 位身份证证正则匹配表达式为:

'^[1-9]\d{5}\d{2}(0[1-9]|(1[0-2]))(([0-2][1-9])|([1-2]0)|31)\d{3}$'
//可简化为:
'^[1-9]\d{7}(0[1-9]|1[0-2])([0-2][1-9]|[1-2]0|31)\d{3}$'

PHP 中校验为:

const ID_15_PREG = '/^[1-9]\d{7}(0[1-9]|1[0-2])([0-2][1-9]|[1-2]0|31)\d{3}$/';

public static function validate($id)
{
    if (!is_string($id) || empty($id)) {
        return false;
    } else if (strlen($id) == 15 && preg_match(static::ID_15_PREG, $id)) {
        return true;
    }
    return false;
}

18位

同理,18 位身份证DDDDDD YYYYMMDD XXX Y的每部分的正则匹配表达式为:

部分名 正则表达式
DDDDDD [1-9]\d{5}
YYYYMMDD ([1-9]\d{3})(0[1-9]|(1[0-2]))(([0-2][1-9])|([1-2]0)|31)
XXX \d{3}
Y \d

由此可得 18 位身份证证正则匹配表达式为:

'^[1-9]\d{5}([1-9]\d{3})((0[1-9]|(1[0-2]))(([0-2][1-9])|([1-2]0)|31)\d{3}\d|[Xx]$'
//可简化为:
'^[1-9]\d{5}[1-9]\d{3}(0[1-9]|1[0-2])([0-2][1-9]|[1-2]0|31)(\d{4}|\d{3}[Xx])$'

根据校验位校验规则,实现 校验位 的编码:

public static function getCheckBit($id)
{
    if (18 !== strlen($id)) {
        return false;
    }
    $yArr = ['1', '0', 'X', '9', '8', '7', '6', '5', '4', '3', '2'];
    $wArr = [7, 9, 10, 5, 8, 4, 2, 1, 6, 3, 7, 9, 10, 5, 8, 4, 2];
    $sum = 0;
    for ($i = strlen($id)-2; $i>=0; $i--) {
        $sum += $id[$i] * $wArr[$i];
    }
    $key = $sum % 11;
    return $yArr[$key];
}

所以,PHP 中校验逻辑为:

const ID_18_PREG = '/^[1-9]\d{5}[1-9]\d{3}(0[1-9]|1[0-2])([0-2][1-9]|[1-2]0|31)(\d{4}|\d{3}[Xx])$/';

public static function validate($id)
{
    if (!is_string($id) || empty($id)) {
        return false;
    } else if (strlen($id) == 18 && preg_match(static::ID_18_PREG, $id) && strtoupper($id[17]) === self::getCheckBit($id)) {
       return true;
    } else if (strlen($id) == 15 && preg_match(static::ID_15_PREG, $id)) {
        return true;
    }
    return false;
}

15位转化为18位

在金融等某些特殊行业,需要将 15 位身份证号码格式化为 18 位。由于 15 位身份证颁发年份都是 19** 年,所以在转化为 18 位时补充出生年份时直接添加 19 即可。

转化步骤:

  • 年份补全成 4 位,年份前直接添加 19;
  • 补全上步的新号码为 18 位,可以在原号码末尾直接追加 X;
  • 计算新号码的校验位并替换原校验位值;

15 位身份证转化为 18 位的代码如下:

public static function format18($id)
{
    if (!static::validate($id)) {
        return '';
    } else if (15 !== strlen($id)) {
        return $id;
    }
    $newId = substr($id, 0, 6) . '19' . substr($id, -9) . 'X';
    $newId[17] = static::getCheckBit($newId);
    return $newId;
}

转化示例结果:

//15位---------------------18位
'370725881105149' => '37072519881105149X'

你可能感兴趣的:(身份证的编码规则)