验证身份证号的正确性

1.需求:

填写人员信息时,输入的身份证号保证不会输入错误,对输入的身份证号进行校验

身份证校验  
如果让你设计个程序,用什么变量保存身份证号码呢?长整数可以吗?不可以! 
 因为有人的身份证最后一位是 "X"  实际上,除了最后一位的X,不会出现其它字母!  身份证号码18位 = 17位 校验码

2.实现的算法:ISO 7064:1983.MOD11-2校验码计算法

3.思路:

很多游戏账号申请,或者某些网站注册时需要填身份证号,当输入错误的身份证号时会提示出错,感觉好像真能识别身份证信息一样,其实不然。

实质上由于其没有权限接入公安系统,它只能根据最后一位校验位来判断该身份证号码是否有效,为了不泄露个人隐私,而又可以轻松的拿到账号,可以伪造身份证号,前17为自己随机伪造,只需要计算出最后一位校验位即可。

ISO 7064:1983.MOD11-2校验码计算法 : (身份证校验码-18)

假设某一17位数字是

17位数字 1 2 3 4 5 6 7 8 9 0 1 2 3 4 5 6 7

加权因子 7 9 10 5 8 4 2 1 6 3 7 9 10 5 8 4 2

计算17位数字各位数字与对应的加权因子的乘积的和S1×72×93×104×55×86×47×28×19×60×31×72×93×104×55×86×47×2368

计算S÷11的余数T368 mod 115

(( 余数010对应校验码为[1, 0, X , 9, 8, 7, 6, 5, 4, 3, 2],算法如下 ))

计算(12-T÷11的余数R,如果R10,校验码为字母“X”;如果R≠10,校验码为数字“R”:(12-5mod 117

17位数字的校验码就是7,聚合在一为123456789012345677

使用以上算法计算一下自己身份证最后一位,即可知道算法正确与否。


4.身份证号验证工具类

public class IDCardUtil {
final static Map<Integer, String> zoneNum = new HashMap<>(); 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(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 certNo 号码内容 *@return 是否有效 null和"" 都是false */ public static boolean isIDCard(String certNo) { if (certNo == null || (certNo.length() != 15 && certNo.length() != 18)) return false; final char[] cs = certNo.toUpperCase().toCharArray(); //校验位数 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]; } } //校验区位码 if (!zoneNum.containsKey(Integer.valueOf(certNo.substring(0, 2)))) { return false; } //校验年份 String year = certNo.length() == 15 ? getIdcardCalendar() + certNo.substring(6, 8) : certNo.substring(6, 10); final int iyear = Integer.parseInt(year); if (iyear < 1900 || iyear > Calendar.getInstance().get(Calendar.YEAR)) return false;//1900年的PASS,超过今年的PASS //校验月份 String month = certNo.length() == 15 ? certNo.substring(8, 10) : certNo.substring(10, 12); final int imonth = Integer.parseInt(month); if (imonth < 1 || imonth > 12) { return false; } //校验天数 String day = certNo.length() == 15 ? certNo.substring(10, 12) : certNo.substring(12, 14); final int iday = Integer.parseInt(day); if (iday < 1 || iday > 31) return false; //校验"校验码" return certNo.length() == 15 || cs[cs.length - 1] == PARITYBIT[power % 11]; } private static int getIdcardCalendar() { GregorianCalendar curDay = new GregorianCalendar(); int curYear = curDay.get(Calendar.YEAR); return Integer.parseInt(String.valueOf(curYear).substring(2)); } /** 特别注意: ”Ⅹ“ 是 罗马数字 10, 不是 英文大写字母 ”X“, 此处为了编码方便,使用了英文字母 ‘X’ 代替; */ }

 

转载于:https://www.cnblogs.com/inspred/p/8563271.html

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值