关于×××号的那些事

转自:http://www.biaodianfu.com/id-number.htmlhtml


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

1、×××号码的组成ide

一、地址码(×××号码前六位)post

表示编码对象常住户口所在县(市、镇、区)的行政区划代码。1-2位省、自治区、直辖市代码; 3-4位地级市、盟、自治州代码; 5-6位县、县级市、区代码。网站

关于行政区划代码具体的对应关系,能够从×××网站上直接获取。编码

具体地址为:http://www.stats.gov.cn/tjbz/xzqhdm/code

二、生日期码(×××号码第七位到第十四位)htm

表示编码对象出生的年、月、日,其中年份用四位数字表示,年、月、日之间不用分隔符。例如:1981年05月11日就用19810511表示。对象

三、顺序码(×××号码第十五位到十七位)get

地址码所标识的区域范围内,对同年、月、日出生的人员编定的顺序号。其中第十七位奇数分给男性,偶数分给女性。

四、校验码(×××号码最后一位)

是根据前面十七位数字码,按照ISO 7064:1983.MOD 11-2校验码计算出来的检验码。做为尾号的校验码,是由号码编制单位按统一的公式计算出来的,若是某人的尾号是0-9,都不会出现X,但若是尾号是10,那么就得用X来代替。

2、×××校验码计算方法

  1. 将前面的×××号码17位数分别乘以不一样的系数。从第一位到第十七位的系数分别为:7-9-10-5-8-4-2-1-6-3-7-9-10-5-8-4-2。

  2. 将这17位数字和系数相乘的结果相加。

  3. 用加出来和除以11,看余数是多少?

  4. 余数只可能有0-1-2-3-4-5-6-7-8-9-10这11个数字。其分别对应的最后一位×××的号码为1-0-X -9-8-7-6-5-4-3-2。经过上面得知若是余数是2,×××的最后一位号码就是罗马数字x。若是余数是10,就会在×××的第18位数字上出现的是2。

3、×××校验Python版


#-*- coding:UTF-8 -*-
import re
from datetime import datetime
              
__all__ = ["GeneratIdentityChecker"]
              
class ExceptionIdentityChecker(Exception):
    pass
              
def GeneratIdentityChecker(country = "China"):
    cls = globals().get(country+"IdentityChecker")
    if cls:
        return cls()
    else:
        raise ExceptionIdentityChecker(country+" identity's checker not found")
              
class ChinaIdentityChecker():
    def __init__(self):
        self.anWi = (7,9,10,5,8,4,2,1,6,3,7,9,10,5,8,4,2)
        self.cnAreaPart = 6
        self.cnMod = 11
        self.csYearPre = "19"
        self.csCheckCode = "10X98765432"
        self.cnMinArea = 150000
        self.cnMaxArea = 700000
              
    def check(self,code,**options):
        code = code.upper()
        if len(code) not in (15,18) or not re.match(r"^\d{15}$|^\d{17}[\dxX]$", code):
            return False
              
        area = int(code[0:self.cnAreaPart])
              
        if not (self.cnMinArea <= area <= self.cnMaxArea):
            return False
              
        birthday = self.csYearPre+code[6:12] if len(code) == 15 else code[6:14]
        try:
            datetime.strptime(birthday, "%Y%M%d")
        except:
            return False
              
        if len(code) == 18:
            wi = self.anWi
            total = 0
            for i in range(16,-1,-1):
                total += int(code[i])*wi[i]
            if not self.csCheckCode[total%self.cnMod] == code[17]:
                return False
              
        for i in options:
            try:
                checker = getattr(self,"_check_"+i)
            except:
                return False
              
            if not checker(code,options[i]):
                return False
              
        return True
              
    def _check_sex(self,code,sex):
        sex_flag = code[14] if len(code) == 15 else code[16]
        sex = sex.upper()
        if sex == "M":
            return int(sex_flag)%2 == 1
        elif sex == "F":
            return int(sex_flag)%2 == 0
        else:
            return False
              
if __name__ == "__main__":
    gic = GeneratIdentityChecker()
    print gic.check("34052419800101001X")