本文将使用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
IDValidator.PHP
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'],6 );
$birth = $code ['type'] === 18 ? substr ( $code ['body'],6,8 ) :
substr ( $code ['body'],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'],6);
$birth = ($code['type'] === 18 ? substr($code['body'],8) :
substr($code['body'],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,4 ) . '-' . substr ( $birth,4,2 ) . '-' . substr ( $birth,- 2 )) : ('19' . substr ( $birth,2,- 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),'0');
$city = self::$util->str_pad(self::$util->rand(20),'0');
$area = self::$util->str_pad(self::$util->rand(20),'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),'0');
$mo = self::$util->str_pad(self::$util->rand(12,1),'0');
$da = self::$util->str_pad(self::$util->rand(28,'0');
if ($isFifteen) {
return $addr . $yr . $mo . $da
. self::$util->str_pad(self::$util->rand(999,3,'1');
}
$yr = '19' . $yr;
$body = $addr . $yr . $mo . $da . self::$util->str_pad(self::$util->rand(999,'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);
}
}
$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"));