当前位置: 首页 > news >正文

Golang 身份证号码校验

文章目录

  • 1.简介
  • 2.构成规则
    • 行政区划代码
    • 出生日期码
    • 顺序码
    • 校验码
    • 示例
  • 3.Golang 实现校验
  • 4.dablelv/cyan
  • 5.小结
  • 参考文献

1.简介

公民身份号码是中华人民共和国为每个公民从出生之日起编定的唯一的、终身不变的身份代码。

公民身份号码在中华人民共和国公民办理涉及政治、经济、社会生活等权益事务方面广泛使用,由中华人民共和国公安部负责公民身份号码的编制和组织实施工作。

早期身份号码叫社会保障号,为 15 位。

1999年8月26日中华人民共和国国务院发布《国务院关于实行公民身份号码制度的决定》(国发[1999]15号),这个文件规定自1999年10月1日起在全国建立和实行公民身份号码制度。

这个公民身份号码就是我们现在使用的第二代身份证号码,为18位,且终身不变。

在这里插入图片描述

2.构成规则

公民身份号码是特征组合码。

公民身份号码按照 GB11643-1999《公民身份号码》国家标准编制,由18位数字组成:前6位为行政区划代码,第7至14位为出生日期码,第15至17位为顺序码,第18位为校验码。

18位数字组合的方式是:
在这里插入图片描述

行政区划代码

对于内地户籍居民,地址码是公民首次获得身份号码(例如新生儿出生登记、无户口人员登记户口)时所在县(市、镇、区)的行政区划代码,如110102是北京市西城区,如果日后行政区划出现调整或将户口迁往外地地址码也不会改变。由于新生儿通常根据属人主义确定户籍,故地址码并不总能代表公民的出生地。

对于持有港澳台居民居住证的港澳台居民,行政区划代码根据原住地分配台湾(71)、香港(81)和澳门(82),只精确到省级,第三位至第六位全部为 0。因此,台湾、香港和澳门居民的地址码分别为 710000、810000 和 820000。

出生日期码

出生日期码表示公民出生的公历日期:年(4位)、月(2位)、日(2位)。

顺序码

顺序码是给同行政区划代码同出生日期码的人编定的顺序号,其中单数分配给男性,双数分配给女性

校验码

校验码采用的是 ISO 7064:1983, MOD 11-2 校验码系统。校验码为一位数,但如果最后采用校验码系统计算的校验码是“10”,碍于身份证号码为18位的规定,则以“X”代替校验码“10”。

校验码计算方法如下:

  1. 将身份证号码前17位数分别乘以不同的系数。

从第一位到第十七位的系数分别为:

7-9-10-5-8-4-2-1-6-3-7-9-10-5-8-4-2
  1. 将这17位数字和系数相乘的结果相加。

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

  3. 余数只可能有 11个 数字:

0-1-2-3-4-5-6-7-8-9-10

分别对应的最后一位身份号码为:

1-0-X-9-8-7-6-5-4-3-2

示例

例如,假设有一名女性,出生地为北京市西城区(对应地址码为110102),出生于1984年4月6日(对应出生日期码为19840406),登记时的顺序码为970(女性分配为双数,男性为单数),则校验码为X,完整的公民身份号码为11010219840406970X。

如果这名女性在香港出生,并同时持有香港永久性居民身份证,那么在申请港澳台居民居住证时,其对应地址码就为810000,又因为出生于1994年8月23日(对应出生日期码为19940823),登记时的顺序码为002(女性分配为双数,男性为单数),则校验码为1,完整的公民身份号码为 810000199408230021。

3.Golang 实现校验

下面以 Golang 为例,给出实现。

// Chinese id card number.
const idCodeRegexString = "^[1-9]\\d{5}(19|20)\\d{2}(0[1-9]|1[0-2])(0[1-9]|[12]\\d|3[01])\\d{3}[\\dX]$"var	idCodeRegex  = regexp.MustCompile(idCodeRegexString)// IsChineseIdCode checks string is chinese id card number.
func IsChineseIdCode(input string) bool {if len(input) != 18 {return false}// Check basic format.if !idCodeRegex.MatchString(input) {return false}// Check if the birthday is valid.birthday := input[6:14]_, err := time.Parse("20060102", birthday)if err != nil {return false}// Calculate the checksum.weights := []int{7, 9, 10, 5, 8, 4, 2, 1, 6, 3, 7, 9, 10, 5, 8, 4, 2}checkCode := []byte{'1', '0', 'X', '9', '8', '7', '6', '5', '4', '3', '2'}var sum intfor i := 0; i < 17; i++ {var num intnum, err = strconv.Atoi(string(input[i]))if err != nil {return false}sum += num * weights[i]}mod := sum % 11return checkCode[mod] == input[17]
}

校验示例:

func main() {// empty stringfmt.Println(IsChineseIdCode("")) // false// length incorrectfmt.Println(IsChineseIdCode("44030619810825051")) // false// birthday incorrectfmt.Println(IsChineseIdCode("440306888808250512")) // false// check code incorrectfmt.Println(IsChineseIdCode("440306198108250511")) // false// valid id numberfmt.Println(IsChineseIdCode("440306198108250515")) // false
}

运行输出:

false
false
false
false
true

4.dablelv/cyan

上述校验函数已经添加至工具库 dablelv/cyan,可 import 直接使用,欢迎大家 Star 和 PR。

package mainimport ("fmt""github.com/dablelv/cyan/strings"
)func main() {// length incorrectfmt.Println(strings.IsChineseIdCode("44030619810825051")) // false// birthday incorrectfmt.Println(strings.IsChineseIdCode("440306888808250512")) // false// check code incorrectfmt.Println(strings.IsChineseIdCode("440306198108250511")) // false// valid id numberfmt.Println(strings.IsChineseIdCode("440306198108250515")) // true
}

运行输出:

false
false
false
false
true

5.小结

通过上述代码,我们实现了一个高效的身份证号码校验工具。虽然本地校验可过滤大部分无效号码,但在实际业务中仍需结合公安系统接口进行实名认证。此工具适用于以下场景:

  • 用户注册时的初步格式校验。

  • 数据清洗中的非法号码过滤。

完整代码已考虑边界条件(如闰年、校验码 X 的大小写),开发者可根据需求扩展地址码库或优化正则表达式。

注意事项:身份证号码包含个人敏感信息,处理时需遵守《个人信息保护法》,避免数据泄露。


参考文献

ISO 7064:1983
GB 11643-1999 - 国家标准全文公开
国务院关于实行公民身份号码制度的决定

相关文章:

  • 【优选算法 | 位运算】位运算基础:深入理解二进制操作
  • ASP.NET MVC​ 入门与提高指南七
  • < 自用文 Texas style Smoker > 美式德克萨斯烟熏炉 从设计到实现 (第一部分:烹饪室与燃烧室)
  • 基于Redis实现-UV统计
  • openEuler 22.03 安装 Mysql 5.7,RPM 在线安装
  • 解决 3D Gaussian Splatting 中 SIBR 可视化组件报错 uv_mesh.vert 缺失问题【2025最新版!】
  • 组件通信-v-model
  • [Control-Chaos] Toxic Cascade(毒性級鏈)
  • 青少年编程与数学 02-018 C++数据结构与算法 22课题、并行算法
  • linux的信号量初识
  • 【Vue】Vue与UI框架(Element Plus、Ant Design Vue、Vant)
  • 第一章:A Primer on Memory Consistency and Cache Coherence - 2nd Edition
  • C++之IO流
  • 计算方法实验四 解线性方程组的间接方法
  • 【Unity】使用XLua实现C#访问Lua文件
  • JavaScript常规解密技术解析指南
  • 对称加密算法(AES、ChaCha20和SM4)Python实现——密码学基础(Python出现No module named “Crypto” 解决方案)
  • Linux系统:进程程序替换以及相关exec接口
  • 大学生入学审核系统设计与实现【基于SpringBoot + Vue 前后端分离技术】
  • 记忆翻牌游戏:认知科学与状态机的交响曲
  • 艺术开卷|韩羽读齐白石:妙在似与不似之间
  • AI把野史当信史?警惕公共认知的滑坡
  • 美国季度GDP时隔三年再现负增长,特朗普政府关税政策对美国经济负面影响或将持续
  • 大学2025丨对话深大人工智能学院负责人李坚强:产学研生态比“造天才”更重要
  • 十二届上海市委第六轮巡视全面进驻,巡视组联系方式公布
  • 街区党支部书记们亮出治理实招,解锁“善治街区二十法”