现在的位置: 首页 > 综合 > 正文

【原创】新旧身份证合法性验证及验证算法

2013年02月25日 ⁄ 综合 ⁄ 共 3680字 ⁄ 字号 评论关闭

         以前浏览网站时,偶尔会发现有些网站注册时需要填写身份证,而且对身份证检验相当严格,猜想必定存在某种算法。今天恰好有空,上网找了一下,有些收获,不敢独享,现与众网友共享。本人已将算法用C#实现,希望大家能多多指点。

1.         中国公民身份证常识:

我国现行使用公民身份证号码有两种尊循两个国家标准,〖GB 11643-1989〗和〖GB 11643-1999〗。

GB 11643-1989〗中规定的是15位身份证号码:排列顺序从左至右依次为:六位数字地址码,六位数字出生日期码,三位数字顺序码,其中出生日期码不包含世纪数。

6行政区划分代码

6位出生日期

3位顺序

 

       GB 11643-1999〗中规定的是18位身份证号码:公民身份号码是特征组合码,由十七位数字本体码和一位数字校验码组成。排列顺序从左至右依次为:六位数字地址码,八位数字出生日期码,三位数字顺序码和一位数字校验码。

6位行政区划分代码

6位出生日期

3位顺序码

1位校验码

       行政区划分代码:表示编码对象常住户口所在县(市、旗、区)的行政区划代码。

       行政区划分代码国家标准GB T 2260-1999

       出生日期码:表示编码对象出生的年、月、日,其中年份用四位数字表示,年、月、日之间不用分隔符。

       顺序码:表示同一地址码所标识的区域范围内,对同年、同月、同日出生的人员编定的顺序号。顺序码的奇数分给男性,偶数分给女性。

    校验码:是根据前面十七位数字码,按照ISO 7064:1983.MOD 11-2校验码计算出来的检验码。

2.         算法

关于身份证号码最后一位的校验码的算法如下:

假设最后一位的校验码为R

C=(a[i]*w[i])%11 (i=2,3,…,18)

其中∑:表示求和

i:表示身份证号码每一位的序号,从右至左,最左侧为18,最右侧位1

*:表示乘号

a[i]:表示身份证号码第i位上的号码

w[i]:表示第i位上的权值,计算方法:w[i]=2^(i-1)%11

%:表示求模运算

^:表示求幂运算

经过上述方法得到的C的范围在0~10之间,它与身份证最后一位校验位的对应规则为:

C

0

1

2

3

4

5

6

7

8

9

10

R

1

0

X

9

8

7

6

5

4

3

2

由此看出 X 就是 10,罗马数字中的 10 就是X,所以在新标准的身份证号码中可能含有非数字的字母X

3.         C#实现

using System;
using
 System.Collections.Generic;
using
 System.Text;

namespace Zwf.IdCard
{
    
public class
 Check
    {
        
//位权值数组

        private static byte[] Weight = new byte[17];
        
//身份证行政区划代码部分长度

        private static byte fPart = 6;
        
//算法求模关键参数

        private static byte fMode = 11;
        
//旧身份证长度

        private static byte oIdLen = 15;
        
//新身份证长度

        private static byte nIdLen = 18;
        
//新身份证年份标记值

        private static string yearFlag = "19";
        
//校验字符串

        private static string checkCode = "10X98765432";
        
//最小行政区划分代码

        private static int minCode = 110000;
        
//最大行政区划分代码

        private static int maxCode = 820000;
        
private static Random rand = new
 Random();

        /// <summary>
        
/// 计算位权值数组
        
/// </summary>

        private static void SetWBuffer()
        {
            
for (int i = 0; i < Weight.Length; i++
)
            {
                
int k = (int)Math.Pow(2, (Weight.Length -
 i));
                Weight[i] 
= (byte)(k %
 fMode);
            }
        }

        /// <summary>
        
/// 获取新身份证最后一位校验位
        
/// </summary>

        
/// <param name="idCard">身份证号码</param>
        
/// <returns></returns>
        private static string GetCheckCode(string idCard)
        {
            
int sum = 0
;
            
///进行加权求和计算

            for (int i = 0; i < Weight.Length; i++)
            {
                sum 
+= int.Parse(idCard.Substring(i, 1)) *
 Weight[i];
            }
            
///求模运算得到模值

            byte mode = (byte)(sum % fMode);
            
return checkCode.Substring(mode, 1
);
        }

        /// <summary>
        
/// 检查身份证长度是否合法
        
/// </summary>

        
/// <param name="idCard">身份证号码</param>
        
/// <returns></returns>
        private static bool CheckLen(string idCard)
        {
            
if (idCard.Length == oIdLen || idCard.Length ==
 nIdLen)
            {
                
return true
;
            }
            
return false
;
        }

        /// <summary>
        
/// 验证是否是新身份证
        
/// </summary>

        
/// <param name="idCard">身份证号码</param>
        
/// <returns></returns>
        private static bool IsNew(string idCard)
        {
            
if (idCard.Length ==
 nIdLen)
            {
                
return true
;
            }
            
return false
;
        }

        /// <summary>
        
/// 获取时间串
        
/// </summary>

        
/// <param name="idCard">身份证号码</param>
        
/// <returns></returns>
        private static string GetDate(string idCard)
        {
            
string str = ""
;
            
if
 (IsNew(idCard))
            {
                str 
= idCard.Substring(fPart, 8
);
            }
            
else

            {
                str 
= yearFlag + idCard.Substring(fPart, 6);
            }
            
return

【上篇】
【下篇】

抱歉!评论已关闭.