首先,我们要先了解一下身份证验证的原理与身份证号码结构。
(1)身份证号码结构:17位数字和1位校验码:6位地址码数字,8位生日数字,3位出生时间顺序码,1位校验码。
(2)地址码(前6位):表示编码对象常住户口所在县(市、旗、区)的行政区划代码,按GB/T2260的规定执行。出生日期码(第七位至十四位):表示编码对象出生的年、月、日,按GB/T7408的规定执行,年、月、日代码之间不用分隔符。
(3)顺序码(第十五位至十七位) :表示在同一地址码所标识的区域范围内,对同年、同月、同日出生的人编定的顺序号, 顺序码的奇数分配给男性,偶数分配给女性。
(4)校验码(第十八位数):十七位数字本体码加权求和公式 S = Sum(Ai * Wi), i = 0, , 16 ,先对前17位数字的权求和;Ai:表示第i位置上的身份证号码数字值 Wi:表示第i位置上的加权因子 Wi: 7 9 10 5 8 4 2 1 6 3 7 9 10 5 8 4 2;
(5)计算模 Y = mod(S, 11)
(6)通过模得到对应的校验码 Y: 0 1 2 3 4 5 6 7 8 9 10
(7)校验码: 1 0 X 9 8 7 6 5 4 3 2
身份证校验码生成算法
根 据〖中华人民共和国国家标准 GB 11643-1999〗中有关公民身份号码的规定,公民身份号码是特征组合码,由十七位数字本体码和一位数字校验码组成。排列顺序从左至右依次为:六位数 字地址码,八位数字出生日期码,三位数字顺序码和一位数字校验码。……顺序码的奇数分给男性,偶数分给女性。校验码 是根据前面十七位数字码,按照ISO 7064:1983.MOD 11-2校验码计算出来的检验码。
先引入公式:
∑(a[i]*W[i]) mod 11 ( i = 2, 3, ..., 18 ) (1)
"*" 表示乘号
i--------表示身份证号码每一位的序号,从右至左,最左侧为18,最右侧为1。
a[i]-----表示身份证号码第 i 位上的号码
W[i]-----表示第 i 位上的权值 W[i] = 2^(i-1) mod 11
计算公式 (1) 令结果为 R
根据下表找出 R 对应的校验码即为要求身份证号码的校验码C。
R 0 1 2 3 4 5 6 7 8 9 10
C 1 0 X 9 8 7 6 5 4 3 2
由此看出 X 就是 10,罗马数字中的 10 就是 X
其次,知道了身份证的结构和验证算法后,下面就通过jsp语言实现身份证合法性的验证。
public class IdcardUtil {
final static Map<Integer, String> zoneNum = new HashMap<Integer, String>();
static {
zoneNum.put(11, "北京");
zoneNum.put(12, "天津");
zoneNum.put(13, "河北");
zoneNum.put(14, "山西");
zoneNum.put(15, "内蒙古");
zoneNum.put(21, "辽宁");
zoneNum.put(22, "吉林");
zoneNum.put(23, "黑龙江");
zoneNum.put(31, "上海");
zoneNum.put(32, "江苏");
zoneNum.put(33, "浙江");
zoneNum.put(34, "安徽");
zoneNum.put(35, "福建");
zoneNum.put(36, "江西");
zoneNum.put(37, "山东");
zoneNum.put(41, "河南");
zoneNum.put(42, "湖北");
zoneNum.put(43, "湖南");
zoneNum.put(44, "广东");
zoneNum.put(45, "广西");
zoneNum.put(46, "海南");
zoneNum.put(50, "重庆");
zoneNum.put(51, "四川");
zoneNum.put(52, "贵州");
zoneNum.put(53, "云南");
zoneNum.put(54, "西藏");
zoneNum.put(61, "陕西");
zoneNum.put(62, "甘肃");
zoneNum.put(63, "青海");
zoneNum.put(64, "宁夏");
zoneNum.put(65, "新疆");
zoneNum.put(71, "台湾");
zoneNum.put(81, "香港");
zoneNum.put(82, "澳门");
zoneNum.put(91, "国外");
}
final static int[] PARITYBIT = { '1', '0', 'X', '9', '8', '7', '6', '5', '4', '3', '2' };
final static int[] POWER_LIST = { 7, 9, 10, 5, 8, 4, 2, 1, 6, 3, 7, 9, 10,
5, 8, 4, 2 };
/**
* 身份证号是否基本有效
* @param s
* 号码内容
* @return 是否有效,null和""都是false
*/
public static boolean isIdcard(String s) {
if (s == null || (s.length() != 15 && s.length() != 18))
return false;
final char[] cs = s.toUpperCase().toCharArray();
// (1)校验位数
int power = 0;
for (int i = 0; i < cs.length; i++) {// 循环比正则表达式更快
if (i == cs.length - 1 && cs[i] == 'X')
break;// 最后一位可以是X或者x
if (cs[i] < '0' || cs[i] > '9')
return false;
if (i < cs.length - 1)
power += (cs[i] - '0') * POWER_LIST[i];
}
// (2)校验区位码
if (!zoneNum.containsKey(Integer.valueOf(s.substring(0, 2)))) {
return false;
}
// (3)校验年份
String year = s.length() == 15 ? "19" + s.substring(6, 8) : s.substring(6, 10);
final int iyear = Integer.parseInt(year);
if (iyear < 1900 || iyear > Calendar.getInstance().get(Calendar.YEAR)) {
return false;// 1900年的PASS,超过今年的PASS
}
// (4)校验月份
String month = s.length() == 15 ? s.substring(8, 10) : s.substring(10,12);
final int imonth = Integer.parseInt(month);
if (imonth < 1 || imonth > 12)
return false;
// (5)校验天数
String day = s.length() == 15 ? s.substring(10, 12) : s.substring(12, 14);
final int iday = Integer.parseInt(day);
if (iday < 1 || iday > 31)
return false;
// (6)校验一个合法的年月日
if (!validate(iyear, imonth, iday))
return false;
// (7)校验“校验码”
if (s.length() == 15)
return true;
return cs[cs.length - 1] == PARITYBIT[power % 11];
}
static boolean validate(int year, int month, int day) {
//比如考虑闰月,大小月等
return true;
}
public static void main(String[] args) {
for(int i=0;i<10;i++) {
final String s = "42230219880101100"+i;
System.out.println(s+" --> "+isIdcard(s));
}
}
}