当前位置:文档之家› 18位身份证号码校验位规则

18位身份证号码校验位规则

18位身份证号码校验位规则2009/01/11 15:33根据〖中华人民共和国国家标准 GB 11643-1999〗中有关公民身份号码的规定,公民身份号码是特征组合码,由十七位数字本体码和一位数字校验码组成。排列顺序从左至右依次为:六位数字地址码,八位数字出生日期码,三位数字顺序码和一位数字校验码。

地址码表示编码对象常住户口所在县(市、旗、区)的行政区划代码。生日期码表示编码对象出生的年、月、日,其中年份用四位数字表示,年、月、日之间不用分隔符。顺序码表示同一地址码所标识的区域范围内,对同年、月、日出生的人员编定的顺序号。顺序码的奇数分给男性,偶数分给女性。校验码是根据前面十七位数字码,按照ISO 7064:1983.MOD 11-2校验码计算出来的检验码。下面举例说明该计算方法。

15位的身份证编码首先把出生年扩展为4位,简单的就是增加一个19,但是这对于1900年出生的人不使用(这样的寿星不多了)

某男性公民身份号码本体码为34052419800101001,首先按照公式⑴计算:

∑(ai×Wi)(mod 11)……………………………………(1)

公式(1)中:

i----表示号码字符从右至左包括校验码在内的位置序号;

ai----表示第i位置上的号码字符值;

Wi----示第i位置上的加权因子,其数值依据公式Wi=2(n-1)(mod 11)计算得出。

i 18 17 16 15 14 13 12 11 10 9 8 7 6 5 4 3 2 1

ai 3 4 0 5 2 4 1 9 8 0 0 1 0 1 0 0 1 a1

Wi 7 9 10 5 8 4 2 1 6 3 7 9 10 5 8 4 2 1

ai×Wi 21 36 0 25 16 16 2 9 48 0 0 9 0 5 0 0 2 a1

根据公式(1)进行计算:

∑(ai×Wi) =(21+36+0+25+16+16+2+9+48++0+0+9+0+5+0+0+2) = 189

189 ÷ 11 = 17 + 2/11

∑(ai×Wi)(mod 11) = 2

然后根据计算的结果,从下面的表中查出相应的校验码,其中X表示计算结果为10:

∑(ai×WI)(mod 11) 0 1 2 3 4 5 6 7 8 9 10

校验码字符值ai 1 0 X 9 8 7 6 5 4 3 2

根据上表,查出计算结果为2的校验码为所以该人员的公民身份号码应该为 340524************。
一个校验身份证号码合法的C程序2008年03月25日 星期二 下午 05:05//摘自MyZone,未亲自验证!
#include
#include
#include
#include

int IsDigitBuf(char *sBuf, int nLen)
{
int i;

if (nLen == 0) return 1;
if (nLen > strlen(sBuf)) nLen = strlen(sBuf);

for (i = 0; i < nLen; i++)
if (!isdigit(sBuf[i])) return 0;

return 1;
}

int checkdate(int iYear, int iMonth, int iDay)
{
if (iYear < 0 || iYear > 9999)
return -1;
switch (iMonth)
{
case 1:
case 3:
case 5:
case 7:
case 8:
case 10:
case 12:
if (iDay <= 0 || iDay > 31)
{
return -3;
}
break;
case 4:
case 6:
case 9:

case 11:
if (iDay <= 0 || iDay > 30)
{
return -3;
}
break;
case 2:
if ((iYear % 4 == 0 && iYear % 100 != 0) || iYear % 400 == 0)
{
if (iDay <= 0 || iDay > 29)
{
return -3;
}
}
else
{
if (iDay <= 0 || iDay > 28)
{
return -3;
}
}
break;
default:
return -2;
}
return 0;
}

int CheckStrDate(char *sDate)
{
int iRet;

char sYear[5];
char sMonth[3];
char sDay[3];

memset(sYear, 0, sizeof(sYear));
memset(sMonth, 0, sizeof(sMonth));
memset(sDay, 0, sizeof(sDay));

if (strlen(sDate) != 8)
{
return -1;
}

memcpy(sYear, sDate, 4);
memcpy(sMonth, sDate+4, 2);
memcpy(sDay, sDate+6, 2);

iRet = checkdate(atoi(sYear), atoi(sMonth), atoi(sDay));
if (iRet != 0)
{
return -1;
}

return 0;
}


int main(int argc, char *argv[])
{
int i;
int iRet;
int iWeight[18] = {7, 9, 10, 5, 8, 4, 2, 1, 6, 3, 7, 9, 10, 5, 8, 4, 2, 1};
char cCheck[11] = {'1', '0', 'X', '9', '8', '7', '6', '5', '4', '3', '2'};
char sIdCardNo[20];
char sDate[8 + 1];
int iDate;
int Sum = 0;

if (argc < 2) {
printf("FAIL\n");
exit(1);
}

memset(sIdCardNo, 0, 20);
memcpy(sIdCardNo, argv[1], 18);
if ((strlen(sIdCardNo) == 15) && ((iRet = IsDigitBuf(sIdCardNo, 15)) > 0)) {
printf("OK\n");
exit(0);
} else if (strlen(sIdCardNo) != 18) {
printf("FAIL\n");
exit(1);
}

/* 身份证7-14位是否有效 */
memset(sDate, 0, sizeof(sDate));
memcpy(sDate, sIdCardNo + 6, 4);
iDate = atoi(sDate);
if (iDate < 1900 || iDate > 2020) {
printf("FAIL\n");
exit(1);
}
memset(sDate, 0, sizeof(sDate));
memcpy(sDate, sIdCardNo + 6, 8);
iRet = CheckStrDate(sDate);
if (iRet < 0) {
printf("FAIL\n");
exit(1);
}

/* 身份证18位校验位是否有效 */
for (i = 0; i < 17; i ++) {
memset(sDate, 0, sizeof(sDate));
sDate[0] = sIdCardNo[i];
iDate = atoi(sDate);
Sum += iWeight[i] * iDate;
}
Sum %= 11;
if ('x' == sIdCardNo[17]) {
sIdCardNo[17] = 'X';
}
if (cCheck[Sum] != sIdCardNo[17]) {
printf("FAIL\n");
exit(1);
}

printf("OK\n");
exit(0);

}




相关主题
文本预览
相关文档 最新文档