Golang 身份证号码校验

文章目录

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 为例,给出实现。

go 复制代码
// 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 int
	for i := 0; i < 17; i++ {
		var num int
		num, err = strconv.Atoi(string(input[i]))
		if err != nil {
			return false
		}
		sum += num * weights[i]
	}
	mod := sum % 11
	return checkCode[mod] == input[17]
}

校验示例:

go 复制代码
func main() {
	// empty string
	fmt.Println(IsChineseIdCode("")) // false

	// length incorrect
	fmt.Println(IsChineseIdCode("44030619810825051")) // false

	// birthday incorrect
	fmt.Println(IsChineseIdCode("440306888808250512")) // false

	// check code incorrect
	fmt.Println(IsChineseIdCode("440306198108250511")) // false

	// valid id number
	fmt.Println(IsChineseIdCode("440306198108250515")) // false
}

运行输出:

复制代码
false
false
false
false
true

4.dablelv/cyan

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

go 复制代码
package main

import (
	"fmt"

	"github.com/dablelv/cyan/strings"
)

func main() {
	// length incorrect
	fmt.Println(strings.IsChineseIdCode("44030619810825051")) // false

	// birthday incorrect
	fmt.Println(strings.IsChineseIdCode("440306888808250512")) // false

	// check code incorrect
	fmt.Println(strings.IsChineseIdCode("440306198108250511")) // false

	// valid id number
	fmt.Println(strings.IsChineseIdCode("440306198108250515")) // true
}

运行输出:

复制代码
false
false
false
false
true

5.小结

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

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

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

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

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


参考文献

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

相关推荐
jiunian_cn19 分钟前
【c++】【STL】queue详解
开发语言·c++·visualstudio
achene_ql40 分钟前
C++ 与 Lua 联合编程
开发语言·c++·lua
钢铁男儿1 小时前
C#编程精要:局部变量、类型推断与常量深度解析
java·开发语言·c#
Charlotte's diary2 小时前
计算机网络 - stp生成树实验
开发语言·计算机网络·php·大作业
weniry2 小时前
动态库与静态库的区别
开发语言·c++
hac13222 小时前
SpringBoot多环境配置
java·spring boot·后端
wacpguo2 小时前
VS Code + Linux 远程开发 go
linux·运维·golang
安冬的码畜日常2 小时前
【玩转 JS 函数式编程_016】DIY 实战:巧用延续传递风格(CPS)重构倒计时特效逻辑
开发语言·前端·javascript·重构·函数式编程·cps风格·延续传递风格
月落霜满天4 小时前
贪心算法求解边界最大数
开发语言·算法
Go高并发架构_王工4 小时前
GoFrame框架深度解析:grpool的优势、最佳实践与踩坑经验
服务器·后端·golang