PHP实现中国公民身份证号码有效性验证
本文将使用Java实现中国公民(15位或者18位)身份证号码的相关验证,功能如下: 身份证号有效性验证 分析详细身份证信息 生成一个虚拟的省份证号码。
身份证号码验证
号码的结构 公民身份号码是特征组合码,由十七位数字本体码和一位校验码组成。排列顺序从左至右依次为:六位数字地址码,八位数字出生日期码,三位数字顺序码和一位数字校验码。
地址码(前六位数)
表示编码对象常住户口所在县(市、旗、区)的行政区划代码,按GB/T2260的规定执行。
出生日期码(第七位至十四位)
表示编码对象出生的年、月、日,按GB/T7408的规定执行,年、月、日代码之间不用分隔符。
顺序码(第十五位至十七位)
表示在同一地址码所标识的区域范围内,对同年、同月、同日出生的人编定的顺序号, 顺序码的奇数分配给男性,偶数分配给女性。
校验码(第十八位数)
(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
(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
<?php namespace com\jdk5\blog\IDValidator; class IDValidator { private static $GB2260; private static $instance; private static $cache = array(); private static $util; function __construct() { if (!class_exists("com\jdk5\blog\IDValidator\GB2260")){ include 'GB2260.php'; } if (!class_exists("com\jdk5\blog\IDValidator\util")){ include 'util.php'; } self::$GB2260 = GB2260::getGB2260 (); self::$util = util::getInstance(); } public static function getInstance() { if (is_null ( self::$instance )) { self::$instance = new IDValidator (); } return self::$instance; } function isValid($id) { $code = self::$util->checkArg ( $id ); if ($code === false) { return false; } // 查询cache if (isset ( self::$cache [ $id ] ) && self::$cache [$id] ['valid'] !== false) { return self::$cache [$id] ['valid']; } else { if (! isset ( self::$cache [ $id ] )) { self::$cache [$id] = array (); } } $addr = substr ( $code ['body'], 0, 6 ); $birth = $code ['type'] === 18 ? substr ( $code ['body'], 6, 8 ) : substr ( $code ['body'], 6, 6 ); $order = substr ( $code ['body'], - 3 ); if (! (self::$util->checkAddr ( $addr ) && self::$util->checkBirth ( $birth ) && self::$util->checkOrder ( $order ))) { self::$cache [$id] ['valid'] = false; return false; } // 15位不含校验码,到此已结束 if ($code ['type'] === 15) { self::$cache [$id] ['valid'] = true; return true; } /* 校验位部分 */ // 位置加权 $posWeight = array (); for($i = 18; $i > 1; $i --) { $wei = self::$util->weight ( $i ); $posWeight [$i] = $wei; } // 累加body部分与位置加权的积 $bodySum = 0; $bodyArr = str_split( $code ['body'] ); for($j = 0; $j < count ( $bodyArr ); $j ++) { $bodySum += (intval ( $bodyArr [$j], 10 ) * $posWeight [18 - $j]); } // 得出校验码 $checkBit = 12 - ($bodySum % 11); if ($checkBit == 10) { $checkBit = 'X'; } else if ($checkBit > 10) { $checkBit = $checkBit % 11; } // 检查校验码 if ($checkBit != $code ['checkBit']) { self::$cache [$id] ['valid'] = false; return false; } else { self::$cache [$id] ['valid'] = true; return true; } } // 分析详细信息 function getInfo ($id) { // 号码必须有效 if ($this->isValid($id) === false) { return false; } // TODO 复用此部分 $code = self::$util->checkArg($id); // 查询cache // 到此时通过isValid已经有了cache记录 if (isset(self::$cache[$id]) && isset(self::$cache[$id]['info'])) { return self::$cache[$id]['info']; } $addr = substr($code['body'], 0, 6); $birth = ($code['type'] === 18 ? substr($code['body'], 6, 8) : substr($code['body'], 6, 6)); $order = substr($code['body'], -3); $info = array(); $info['addrCode'] = $addr; if (self::$GB2260 !== null) { $info['addr'] = self::$util->getAddrInfo($addr); } $info ['birth'] = ($code ['type'] === 18 ? (substr ( $birth, 0, 4 ) . '-' . substr ( $birth, 4, 2 ) . '-' . substr ( $birth, - 2 )) : ('19' . substr ( $birth, 0, 2 ) . '-' . substr ( $birth, 2, 2 ) . '-' . substr ( $birth, - 2 ))); $info['sex'] = ($order % 2 === 0 ? 0 : 1); $info['length'] = $code['type']; if ($code['type'] === 18) { $info['checkBit'] = $code['checkBit']; } // 记录cache self::$cache[$id]['info'] = $info; return $info; } // 仿造一个号 function makeID ($isFifteen=false) { // 地址码 $addr = null; if (self::$GB2260 !== null) { $loopCnt = 0; while ($addr === null) { // 防止死循环 if ($loopCnt > 50) { $addr = 110101; break; } $prov = self::$util->str_pad(self::$util->rand(66), 2, '0'); $city = self::$util->str_pad(self::$util->rand(20), 2, '0'); $area = self::$util->str_pad(self::$util->rand(20), 2, '0'); $addrTest = $prov . $city . $area; if (isset(self::$GB2260[$addrTest])) { $addr = $addrTest; break; } $loopCnt ++; } } else { $addr = 110101; } // 出生年 $yr = self::$util->str_pad(self::$util->rand(99, 50), 2, '0'); $mo = self::$util->str_pad(self::$util->rand(12, 1), 2, '0'); $da = self::$util->str_pad(self::$util->rand(28, 1), 2, '0'); if ($isFifteen) { return $addr . $yr . $mo . $da . self::$util->str_pad(self::$util->rand(999, 1), 3, '1'); } $yr = '19' . $yr; $body = $addr . $yr . $mo . $da . self::$util->str_pad(self::$util->rand(999, 1), 3, '1'); // 位置加权 $posWeight = array(); for ($i = 18; $i > 1; $i--) { $wei = self::$util->weight($i); $posWeight[$i] = $wei; } // 累加body部分与位置加权的积 $bodySum = 0; $bodyArr = str_split($body); for ($j = 0; $j < count($bodyArr); $j++) { $bodySum += (intval($bodyArr[$j], 10) * $posWeight[18 - $j]); } // 得出校验码 $checkBit = 12 - ($bodySum % 11); if ($checkBit == 10) { $checkBit = 'X'; } else if ($checkBit > 10) { $checkBit = $checkBit % 11; } return ($body . $checkBit); } }
调用方式:
<?php header("Content-type: text/html; charset=utf-8"); include 'IDValidator.php'; $v = com\jdk5\blog\IDValidator\IDValidator::getInstance(); //生成一个18位身份证号 $id = $v->makeID(); //获取身份证信息 $info = $v->getInfo($id); var_dump($info); //生成一个15位身份证号 $id = $v->makeID(true); $info = $v->getInfo($id); var_dump($info); //验证身份证号是否正确 var_dump($v->isValid("123456789012345678"));