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

身份证号数字说明(身份证的编码规则)(1)

编码规则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)

参数说明:

例如某个 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 即可。

转化步骤:

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'

,