# 身份证校验
## 身份证编码解析

## 身份证编码校验
### 省级地址码校验
华北:北京11,天津12,河北13,山西14,内蒙古15
东北: 辽宁21,吉林22,黑龙江23
华东: 上海31,江苏32,浙江33,安徽34,福建35,江西36,山东37
华中: 河南41,湖北42,湖南43
华南: 广东44,广西45,海南46
西南: 四川51,贵州52,云南53,西藏54,重庆50
西北: 陕西61,甘肃62,青海63,宁夏64,新疆65
特别:中国台湾71,中国香港81,中国澳门82
#### 校验码
```js
// ^从头开始校验 第一位:1-9 第二位:0-9
var pattern = /^[1-9][0-9]/
```
### 年份码校验
- 年份码长4位
- 以数字18,19或20开头
- 剩余两位为0-9的数字
#### 校验码
```js
// 18/19/20开头 \d数字(0-9)随机出现两位
var year = /(18|19|20)\d{2}/
```
### 月份码校验
- 月份码长2位
- 第一位数字为0,第二位数字为1-9,表示前9个月份
- 或者第一位数字为1,第二位数字为0-2,表示10、11、12三个月份
#### 校验码
```js
// 月份:0[1-9]前九月 |或则 1[0-2]尾三月10、11、12
var year = /((0[1-9])|(1[0-2]))/
```
### 日期校验
- 日期码长2位
- 第一位数字为0-2,第二位数字为1-9
- 或者是10,20,30,31
#### 校验码
```js
// 日期:01-09 | 11-19 | 21-29 | 特殊日期 10、20、30、31
var date = /(([0-2][1-9])|10|20|30|31)/
```
### 顺序码校验
- 顺序码长3位
- 顺序码是数字
#### 校验码
```js
// 顺序码:\d数字 {3}出现3位
var order = /\d{3}/
```
### 校验码校验
- 校验码长1位
- 可以是数字,字母x或字母X
- 校验码计算公式:ai 表示身份证编码地 i 位值,Wi 表示第 i 位的加权因子值 ,根据身份证主体码(前17位)分别与对应的加权因子(加权因子表)计算乘积再求和,根据所得结果与 11 取模得到X值。
- 根据 X 值查询 校验码换算表,得出 a18 即校验码值。

- 加权因子表

- 校验码换算表

## 综合校验
```js
// 校验方法(身份证编码);
var checkCode = function (val) {
// 校验规则 正则表达式
var p = /^[1-9]\d{5}(18|19|20)\d{2}((0[1-9])|(1[0-2]))(([0-2][1-9])|10|20|30|31)\d{3}[0-9Xx]$/;
// 加权因子
var factor = [ 7, 9, 10, 5, 8, 4, 2, 1, 6, 3, 7, 9, 10, 5, 8, 4, 2 ];
// 校验码换算
var parity = [ 1, 0, 'X', 9, 8, 7, 6, 5, 4, 3, 2 ];
// 截取身份证前 17 位编码(除校验码外)
var code = val.substring(17);
// 正则表达式校验
if(p.test(val)) {
// 统计校验和
var sum = 0;
for(var i=0;i<17;i++) {
sum += val[i]*factor[i];
}
// 判断校验码
if(parity[sum % 11] == code.toUpperCase()) {
// 校验成功
return true;
}
}
// 校验失败
return false;
}
// 输出 true, 校验码相符
console.log(checkCode("11010519491231002X"));
// 输出 false, 校验码不符
console.log(checkCode("110105194912310021"));
```
## 例
#### html:
```html
<el-dialog title="用户注册" class="loginForm" :visible.sync="registerFormVisible" width="30%">
<!-- rules:校验名称 ref:校验对象 -->
<el-form :model="registerForm" rules="registerRules" ref="RegisterForm">
<!-- prop:校验项名称 -->
<el-form-item label="身份证号:" prop="srIdCard" :label-width="registerFormLabelWidth">
<el-input v-model="registerForm.srIdCard" autocomplete="off" style="width: 75%"> </el-input>
</el-form-item>
</el-form>
</el-dialog>
```
#### js
```js
export default {
data() {
// 身份证校验
var validateIdCard = (rule, value,callback) => {
// 身份证校验正则表达式
// var p =/^[1-9]\d{5}(18|19|20)\d{2}((0[1-9])|(1[0-2]))(([0-2][1-9])|10|20|30|31)\d{3}[0-9Xx]$/;
// 18 位校验正则表达式
var _IDRe18 =/^([1-6][1-9]|50)\d{4}(18|19|20)\d{2}((0[1-9])|10|11|12)(([0-2][1-9])|10|20|30|31)\d{3}[0-9Xx]$/;
// 15 位校验正则表达式
var _IDre15 =/^([1-6][1-9]|50)\d{4}\d{2}((0[1-9])|10|11|12)(([0-2][1-9])|10|20|30|31)\d{3}$/;
// 加权因子
var factor = [7, 9, 10, 5, 8, 4, 2, 1, 6, 3, 7, 9, 10, 5, 8, 4, 2];
// 校验码换算
var parity = [1, 0, "X", 9, 8, 7, 6, 5, 4, 3, 2];
// 身份证前 17 位
var code = value.substring(17);
// 正则校验
if (_IDre15.test(value)||_IDRe18.test(value)) {
var sum = 0;
for (var i = 0; i < 17; i++) {
sum += value[i] * factor[i];
}
// 校验码校验
if (parity[sum % 11] !== code.toUpperCase()) {
callback(new Error("身份证错误!"));
}
else{
callback();
}
}
else{
callback(new Error("身份证错误!"));
}
};
return{
// 注册校验:relus = registerRules
registerRules: {
// 身份证校验规则
srIdCard: [
// 内置校验
{ required: true, message: "请输入身份证", trigger: "blur" },
// 自定义校验:身份证校验 validateIdCard();
{ validator: validateIdCard, trigger: "blur" },
],
},
// return
},
//data
},
methods: {
// 注册提交
subRegisterForm: function (form) {
// 校验表单
this.$refs[form].validate((valid) => {
// 注册验证成功
if (valid) {
axios
.post(
"http://localhost:8080/user/register",
qs.stringify({
srName: this.registerForm.name,
srPassword: this.registerForm.password,
srIdCard: this.registerForm.srIdCard,
})
)
.then((res) => {
if (res.data.code == 0) {
this.registerFormVisible = false;
this.$message({
type: "success",
message: res.data.msg,
});
return;
}
// 注册失败
this.$message({
type: "success",
message: res.data.msg,
});
});
}
});
},
// method
}
```