文章目录
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"。
校验码计算方法如下:
- 将身份证号码前17位数分别乘以不同的系数。
从第一位到第十七位的系数分别为:
7-9-10-5-8-4-2-1-6-3-7-9-10-5-8-4-2
-
将这17位数字和系数相乘的结果相加。
-
用加出来和除以11,看余数是多少?
-
余数只可能有 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 的大小写),开发者可根据需求扩展地址码库或优化正则表达式。
注意事项:身份证号码包含个人敏感信息,处理时需遵守《个人信息保护法》,避免数据泄露。