身份证信息校验说明[C++Builder]
由于编写系统时涉及到身份证的正确校验从网上搜到身份证号码的检验公式,由于网上C++Builder的源代码很少,特将源代码公布,用Borland C++ builder 6.0 测试通过。<Jason -http://blog.csdn.net/jackson>
一、说明
/*==验证身份证的最后一位检验码======================================+/
根据〖中华人民共和国国家标准GB11643-1999〗中有关公民身份号码的
规定,公民身份号码是特征组合码,由十七位数字本码和一位数字校验
码组成。排列顺序从左至右依次为:六位数字地址区位码,八位数字出
生日期码,三位数字顺序码和一位数字校验码。
地址区位码:表示居民常住户口所在县(市、区)的行政区划代码。
出生日期码:表示居民出生的年、月、日,其中年份用四位数字表示,
年、月、日之间不用分隔符。
数字顺序码:表示同一地址区位码所标识的区域范围内,对同年同月同
日出生的人员编定的顺序号。奇数分给男性,偶数分给女性。
数字校验码:根据前面十七位数字码,按照ISO 7064:1983.MOD 11-2校
验码计算出来的检验码。由(1,2,3,4,5,6,7,8,9,X)构成。
下面举例说明该计算方法并用代码实现。
公式:Ax=(∑(Ai×Wi))(mod 11)…………………………(1)
i----表示身份证号码字符从右至左包括校验码在内的位置序号;
Ai----表示第i位置上的数字号码;
Ax----数字校验码的值;
Wi----示第i位置上的加权因子;
公式:Wi=(2^(i-1))(mod 11)…………………………(2)
设某男性公民身份号码为51310119820113001,首先得到各个变量:
i18 17 16 15 14 13 12 11 10 9 8 7 6 5 4 3 2 1
Ai5 1 3 1 0 1 1 9 8 2 0 1 1 3 0 0 1 Ax
由公式(2)可得:
Wi7 9 10 5 8 4 2 1 6 3 7 9 10 5 8 4 2 1
Ai×Wi 35 9 30 5 0 4 2 9 48 6 0 9 10 15 0 0 2 A1
按照公式(1)计算:
∑(Ai×Wi)=(35+9+30+5+0+4+2+9+48+6+0+9+10+15+0+0+2)=184
184÷11=16+8/11
∑(Ai×Wi)(mod 11)=8
然后根据计算的结果,从下表中查出相应的校验码,X表示结果为10:
∑(Ai×WI)(mod 11)0 1 2 3 4 5 6 7 8 9 10
校验码字符值Ai1 0 X 9 8 7 6 5 4 3 2
根据上面的方法,查出计算结果为8的校验码为4所以该人员的公民身份
号码应该为513101198201130014。
二、代码
//---------------------------------------------------------------------------
bool __fastcall TCLSIDCARD::DepartIDNum(AnsiString p_IDCardStr, int p_iIDMode, AnsiString &p_DistrctCode,
AnsiString &BirDayStr, AnsiString &Gender)
{
//从身份证号码识别第几代身份证、地区代码、出生年月日、性别
int iGender;
p_DistrctCode = "";
BirDayStr = "";
Gender = "";
switch(p_iIDMode) {
case ID_CardGend1: //1 p_DistrctCode = p_IDCardStr.SubString(1,6);
BirDayStr = "19" + p_IDCardStr.SubString(7,2) + "-"+p_IDCardStr.SubString(9,2)+ "-" + p_IDCardStr.SubString(11,2);
iGender = p_IDCardStr.SubString(ID_CardGend1_Len,1).ToInt();//ID_CardGend1_Len 为15
if((iGender % 2) == 1)
G_Gender = "男";
else
G_Gender = "女";
break;
case ID_CARDGend2://2 p_DistrctCode = p_IDCardStr.SubString(1,6);
BirDayStr = p_IDCardStr.SubString(7,4) + "-" + p_IDCardStr.SubString(11,2)+
"-" + p_IDCardStr.SubString(13,2);
iGender = p_IDCardStr.SubString(ID_CARDGend2_Len - 1,1).ToInt();//ID_CARDGend2_Len 为18
if((iGender % 2) == 1)
G_Gender = "男";
else
G_Gender = "女";
break;
default:
break;
}
try{
StrToDate(BirDayStr);
}
catch(...){
Application->MessageBoxA("身份证号出生年月日错误!", "提示",MB_OK +MB_ICONWARNING);
return false;
}
return true;
}
//---------------------------------------------------------------------------
AnsiString __fastcall TCLSIDCARD::Make15To18(AnsiString p_IDCardStr)
{
//由15位身份证号升到18位身份证号的算法函数
int Wi[] = {7, 9, 10, 5, 8, 4, 2, 1, 6, 3, 7, 9, 10, 5, 8, 4, 2, 1};
AnsiString Ai[] = {"1", "0", "X", "9", "8", "7", "6", "5", "4", "3", "2"};
int iTotal = 0;
int iIDNum, iLen;
AnsiString v_IDCard = p_IDCardStr.SubString(1,6) + "19" + p_IDCardStr.SubString(7,ID_CardGend1_Len - 6);
iLen = v_IDCard.Length();
for(int i = 0; i < iLen; i ++) {
iIDNum = v_IDCard.SubString(i + 1, 1).ToInt();
iTotal += iIDNum * Wi[i];
}
int modVal = iTotal % 11;
AnsiString LastNumber = Ai[modVal];
return v_IDCard + LastNumber;
}
//---------------------------------------------------------------------------
bool __fastcall TCLSIDCARD::VerifyIDCARDGend2(AnsiString p_IDCardStr)
{
//校验二代身份证是否有效,最后一位为检验码
int Wi[] = {7, 9, 10, 5, 8, 4, 2, 1, 6, 3, 7, 9, 10, 5, 8, 4, 2, 1};
AnsiString Ai[] = {"1", "0", "X", "9", "8", "7", "6", "5", "4", "3", "2"};
int iTotal = 0;
int iIDNum, iLen;
iLen = p_IDCardStr.Length();
for(int i = 0; i < iLen -1; i ++) {
iIDNum = p_IDCardStr.SubString(i + 1, 1).ToInt();
iTotal += iIDNum * Wi[i];
}
int modVal = iTotal % 11;
AnsiString LastNumber = Ai[modVal];
if (p_IDCardStr.SubString(iLen, 1) == LastNumber)
return true;
else {
Application->MessageBoxA("二代新份证校验有误,请检查身份证号码!", "提示",MB_OK +MB_ICONWARNING);
return false;
}
}
<-Jason 2007.1.3>
本文来自CSDN博客,转载请标明出处:http://blog.csdn.net/JacksonLiang/archive/2007/01/03/1473026.aspx