Php实现中国公民身份证号码有效性验证

  • 发表于
  • PHP

本文将使用Java实现中国公民(15位或者18位)身份证号码的相关验证,功能如下:

  • 身份证号有效性验证
  • 分析详细身份证信息
  • 生成一个虚拟的省份证号码。

身份证号码验证

  1. 号码的结构 公民身份号码是特征组合码,由十七位数字本体码和一位校验码组成。排列顺序从左至右依次为:六位数字地址码,八位数字出生日期码,三位数字顺序码和一位数字校验码。
  2. 地址码(前六位数)
    表示编码对象常住户口所在县(市、旗、区)的行政区划代码,按GB/T2260的规定执行。
  3. 出生日期码(第七位至十四位)
    表示编码对象出生的年、月、日,按GB/T7408的规定执行,年、月、日代码之间不用分隔符。
  4. 顺序码(第十五位至十七位)
    表示在同一地址码所标识的区域范围内,对同年、同月、同日出生的人编定的顺序号, 顺序码的奇数分配给男性,偶数分配给女性。
  5. 校验码(第十八位数)
    (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
//http://blog.jdk5.com/zh/php-chinese-personal-id-card-validation/
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"));

源码下载

相关

Js实现中国公民身份证号码有效性验证