身份证验证

今天准备做一个JS的身份证验证,所以去网上了解了下身份证如何去验证。感觉这小小身份证也有蛮多的学问在里面,所以留下来,备用咯。
本文理论转载于:http://xining.iteye.com/blog/512846
code转载于:http://www.lre.cn/article/Article.asp?ID=858

【身份证号码的规则】
1、15位身份证号码组成:
ddddddyymmddxxs共15位,其中:
dddddd为6位的地方代码,根据这6位可以获得该身份证号所在地。
yy为2位的年份代码,是身份证持有人的出身年份。
mm为2位的月份代码,是身份证持有人的出身月份。
dd为2位的日期代码,是身份证持有人的出身日。
这6位在一起组成了身份证持有人的出生日期。
xx为2位的顺序码,这个是随机数。
s为1位的性别代码,奇数代表男性,偶数代表女性。

2、18位身份证号码组成:
ddddddyyyymmddxxsp共18位,其中:
其他部分都和15位的相同。年份代码由原来的2位升级到4位。最后一位为校验位。
校验规则是:
(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

也就是说,如果得到余数为1则最后的校验位p应该为对应的0.如果校验位不是,则该身份证号码不正确。以下为js版本的校验实例。


function checkCard(card) {
var Errors=new Array(
"验证通过!",
"身份证号码位数不对!",
"身份证号码出生日期超出范围或含有非法字符!",
"身份证号码校验错误!",
"身份证地区非法!");

// 得到地区集合数组
var area = getArea();
var card_array=card.split("");
// s为1位的性别代码,奇数代表男性,偶数代表女性

var ereg, JYM, S, Y;
// 验证身份证地区是否合法
if (area[card.substring(0, 2)] == null)
return Errors[4];
switch (card.length) {
// 15位身份证验证
case 15:
if ((parseInt(card.substr(6, 2)) + 1900) % 4 == 0
|| ((parseInt(card.substr(6, 2)) + 1900) % 100 == 0 && (parseInt(card
.substr(6, 2)) + 1900) % 4 == 0)) {
ereg = /^[1-9][0-9]{5}[0-9]{2}((01|03|05|07|08|10|12)(0[1-9]|[1-2] [0-9]|3[0-1])|(04|06|09|11)(0[1-9]|[1-2][0-9]|30)|02(0[1-9]|[1-2][0-9])) [0-9]{3}$/;// 测试出生日期的合法性
} else {
ereg = /^[1-9][0-9]{5}[0-9]{2} ((01|03|05|07|08|10|12)(0[1-9]|[1-2][0-9]|3[0-1])|(04|06|09|11)(0[1-9]| [1-2][0-9]|30)|02(0[1-9]|1[0-9]|2[0-8]))[0-9]{3}$/;// 测试出生日期的合法性
}
if (ereg.test(card))
return Errors[0];
else
return Errors[2];
break;
// 18位身份证验证
case 18:
// 18位身份号码检测
// 出生日期的合法性检查
// 闰年月日:((01|03|05|07|08|10|12)(0[1-9]|[1-2][0-9]|3[0-1])|
// (04|06|09|11)(0[1-9]|[1-2][0-9]|30)|02(0[1-9]|[1-2][0-9]))
// 平年月日:
// ((01|03|05|07|08|10|12)(0[1-9]|[1-2][0-9]|3[0-1])|(04|06|09|11)(0[1-9]|[1-2][0-9]|30)|02(0[1-9]|1[0-9]|2[0-8]))
if (parseInt(card.substr(6, 4)) % 4 == 0
|| (parseInt(card.substr(6, 4)) % 100 == 0 && parseInt(card
.substr(6, 4)) % 4 == 0)) {
ereg = /^[1-9][0-9]{5}19[0-9]{2}((01|03|05|07|08|10|12)(0[1-9]|[1-2][0-9]|3[0-1])|(04|06|09|11)(0[1-9]|[1-2][0-9]|30)|02(0[1-9]|[1-2][0-9]))[0-9]{3}[0-9Xx]$/;// 闰年出生日期的合法性正则表达式
} else {
ereg = /^[1-9][0-9]{5}19[0-9]{2}((01|03|05|07|08|10|12)(0[1-9]|[1-2][0-9]|3[0-1])|(04|06|09|11)(0[1-9]|[1-2][0-9]|30)|02(0[1-9]|1[0-9]|2[0-8]))[0-9]{3}[0-9Xx]$/;// 平年出生日期的合法性正则表达式
}
if (ereg.test(card)) {// 测试出生日期的合法性
// 计算校验位
S = (parseInt(card_array[0]) + parseInt(card_array[10])) * 7
+ (parseInt(card_array[1]) + parseInt(card_array[11]))
* 9
+ (parseInt(card_array[2]) + parseInt(card_array[12]))
* 10
+ (parseInt(card_array[3]) + parseInt(card_array[13]))
* 5
+ (parseInt(card_array[4]) + parseInt(card_array[14]))
* 8
+ (parseInt(card_array[5]) + parseInt(card_array[15]))
* 4
+ (parseInt(card_array[6]) + parseInt(card_array[16]))
* 2 + parseInt(card_array[7]) * 1
+ parseInt(card_array[8]) * 6 + parseInt(card_array[9])
* 3;
Y = S % 11;
M = "F";
JYM = "10X98765432";
M = JYM.substr(Y, 1);// 判断校验位
if (M == card_array[17])
return Errors[0]; // 检测ID的校验位
else
return Errors[3];
} else
return Errors[2];
break;
break;
}
return Errors[1];
}
function getArea() {
var area = {
11 :"北京",
12 :"天津",
13 :"河北",
14 :"山西",
15 :"内蒙古",
21 :"辽宁",
22 :"吉林",
23 :"黑龙江",
31 :"上海",
32 :"江苏",
33 :"浙江",
34 :"安徽",
35 :"福建",
36 :"江西",
37 :"山东",
41 :"河南",
42 :"湖北",
43 :"湖南",
44 :"广东",
45 :"广西",
46 :"海南",
50 :"重庆",
51 :"四川",
52 :"贵州",
53 :"云南",
54 :"西藏",
61 :"陕西",
62 :"甘肃",
63 :"青海",
64 :"宁夏",
65 :"新疆",
71 :"台湾",
81 :"香港",
82 :"澳门",
91 :"国外"
}
return area;
}
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值