/id-number-util

使用Python随机生成身份证号码及校验、识别、扫描

Primary LanguagePythonApache License 2.0Apache-2.0

身份组成方式

中华人民共和国国家标准GB 11643-1999《公民身份号码》中规定:公民身份号码是特征组合码,由十七位数字本体码和一位校验码组成。

18位数字组合的方式是:

1 1 0 1 0 2 Y Y Y Y M M D D 8 8 8 X
区域码(6位) 出生日期码(8位) 顺序码(2位) 性别码(1位) 校验码(1位)
  • 区域码 指的是公民常住户口所在县(市、镇、区)的行政区划代码,如110102是北京市-西城区。但港澳台地区居民的身份号码只精确到省级。
  • 出生日期码 表示公民出生的公历年(4位)、月(2位)、日(2位)。
  • 顺序码 表示在同一区域码所标识的区域范围内,对同年、同月、同日出生的人编定的顺序号。
  • 性别码 奇数表示男性,偶数表示女性。
  • 最后一位是校验码,这里采用的是ISO 7064:1983,MOD 11-2校验码系统。校验码为一位数,但如果最后采用校验码系统计算的校验码是“10”,碍于身份证号码为18位的规定,则以“X”代替校验码“10”。

校验码计算方法

  • 1. 将身份证号码从右至左标记为 即为校验码;
  • 2. 计算权重系数

所以:

i 18 17 16 15 14 13 12 11 10 9 8 7 6 5 4 3 2 1
Wi 7 9 10 5 8 4 2 1 6 3 7 9 10 5 8 4 2 1
  • 3. 计算
  • 4.

使用Python获取身份证校验码:

def get_check_digit(id_number):
    """ 通过身份证号获取校验码 """
    check_sum = 0
    for i in range(0, 17):
        check_sum += ((1 << (17 - i)) % 11) * int(id_number[i])
    check_digit = (12 - (check_sum % 11)) % 11
    return check_digit if check_digit < 10 else 'X'

随机生成身份证

由上面的组合方式我们可以得出以下代码:

   @classmethod
   def generate_id(cls, sex=0):
       """ 随机生成身份证号,sex = 0表示女性,sex = 1表示男性 """

       # 随机生成一个区域码(6位数)
       id_number = str(random.choice(list(const.AREA_INFO.keys())))
       # 限定出生日期范围(8位数)
       start, end = datetime.strptime("1960-01-01", "%Y-%m-%d"), datetime.strptime("2000-12-30", "%Y-%m-%d")
       birth_days = datetime.strftime(start + timedelta(random.randint(0, (end - start).days + 1)), "%Y%m%d")
       id_number += str(birth_days)
       # 顺序码(2位数)
       id_number += str(random.randint(10, 99))
       # 性别码(1位数)
       id_number += str(random.randrange(sex, 10, step=2))
       # 校验码(1位数)
       return id_number + str(cls(id_number).get_check_digit())

工具类主要功能

if __name__ == '__main__':
    random_sex = random.randint(0, 1)  # 随机生成男(1)或女(0)
    print(IdNumber.generate_id(random_sex))  # 随机生成身份证号
    print(IdNumber('410326199507103197').area_id)  # 地址编码:410326
    print(IdNumber('410326199507103197').get_area_name())  # 地址:河南省洛阳市汝阳县
    print(IdNumber('410326199507103197').get_birthday())  # 生日:1995-7-10
    print(IdNumber('410326199507103197').get_age())  # 年龄:23(岁)
    print(IdNumber('410326199507103197').get_sex())  # 性别:1(男)
    print(IdNumber('410326199507103197').get_check_digit())  # 校验码:7
    print(IdNumber.verify_id('410326199507103198'))  # 检验身份证是否正确:False

[√]: 代码地址: https://github.com/jayknoxqu/id-number-util