身份证校验工具类

   
   
  1. import java.text.ParseException;
  2. import java.text.SimpleDateFormat;
  3. import java.util.Calendar;
  4. import java.util.GregorianCalendar;
  5. import java.util.Hashtable;
  6. import java.util.regex.Matcher;
  7. import java.util.regex.Pattern;
  8. /**
  9. * 身份证号码的格式:610821-20061222-612-X
  10. * 由18位数字组成:前6位为地址码,第7至14位为出生日期码,第15至17位为顺序码,
  11. * 第18位为校验码。检验码分别是0-10共11个数字,当检验码为“10”时,为了保证公民身份证号码18位,所以用“X”表示。虽然校验码为“X”不能更换,但若需全用数字表示,只需将18位公民身份号码转换成15位居民身份证号码,去掉第7至8位和最后1位3个数码。
  12. * (1)前1、2位数字表示:所在省份的代码;
  13. * (2)第3、4位数字表示:所在城市的代码;
  14. * (3)第5、6位数字表示:所在区县的代码;
  15. * (4)第7~14位数字表示:出生年、月、日;
  16. * (5)第15、16位数字表示:所在地的派出所的代码;
  17. * (6)第17位数字表示性别:奇数表示男性,偶数表示女性
  18. * (7)第18位数字是校检码:根据一定算法生成
  19. * @author jiaqi.wang
  20. *
  21. */
  22. public class CheckIDCard {
  23. public static String IDCardValidate(String IDStr) throws ParseException {
  24. String tipInfo = "该身份证有效";// 记录错误信息
  25. String Ai = "";
  26. // 判断号码的长度 15位或18位
  27. if (IDStr.length() != 15 && IDStr.length() != 18) {
  28. tipInfo = "身份证号码长度应该为15位或18位";
  29. return tipInfo;
  30. }
  31. // 18位身份证前17位位数字,如果是15位的身份证则所有号码都为数字
  32. if (IDStr.length() == 18) {
  33. Ai = IDStr.substring(0, 17);
  34. } else if (IDStr.length() == 15) {
  35. Ai = IDStr.substring(0, 6) + "19" + IDStr.substring(6, 15);
  36. }
  37. if (!isNumeric(Ai)) {
  38. tipInfo = "身份证15位号码都应为数字 ; 18位号码除最后一位外,都应为数字";
  39. return tipInfo;
  40. }
  41. // 判断出生年月是否有效
  42. String strYear = Ai.substring(6, 10);// 年份
  43. String strMonth = Ai.substring(10, 12);// 月份
  44. String strDay = Ai.substring(12, 14);// 日期
  45. if (!isDate(strYear + "-" + strMonth + "-" + strDay)) {
  46. tipInfo = "身份证出生日期无效";
  47. return tipInfo;
  48. }
  49. GregorianCalendar gc = new GregorianCalendar();
  50. SimpleDateFormat s = new SimpleDateFormat("yyyy-MM-dd");
  51. try {
  52. if ((gc.get(Calendar.YEAR) - Integer.parseInt(strYear)) > 150
  53. || (gc.getTime().getTime() - s.parse(
  54. strYear + "-" + strMonth + "-" + strDay).getTime()) < 0) {
  55. tipInfo = "身份证生日不在有效范围";
  56. return tipInfo;
  57. }
  58. } catch (NumberFormatException | ParseException e) {
  59. e.printStackTrace();
  60. }
  61. if (Integer.parseInt(strMonth) > 12 || Integer.parseInt(strMonth) == 0) {
  62. tipInfo = "身份证月份无效";
  63. return tipInfo;
  64. }
  65. if (Integer.parseInt(strDay) > 31 || Integer.parseInt(strDay) == 0) {
  66. tipInfo = "身份证日期无效";
  67. return tipInfo;
  68. }
  69. // 判断地区码是否有效
  70. Hashtable<String, String> areacode = GetAreaCode();
  71. //如果身份证前两位的地区码不在Hashtable,则地区码有误
  72. if (areacode.get(Ai.substring(0, 2)) == null) {
  73. tipInfo = "身份证地区编码错误";
  74. return tipInfo;
  75. }
  76. if (!isVarifyCode(Ai, IDStr)) {
  77. tipInfo = "身份证校验码无效,不是合法的身份证号码";
  78. return tipInfo;
  79. }
  80. return tipInfo;
  81. }
  82. /**
  83. * 判断第18位校验码是否正确
  84. * 第18位校验码的计算方式:
  85.   1. 对前17位数字本体码加权求和
  86.   公式为:S = Sum(Ai * Wi), i = 0, ... , 16
  87.   其中Ai表示第i个位置上的身份证号码数字值,Wi表示第i位置上的加权因子,其各位对应的值依次为: 7 9 10 5 8 4 2 1 6 3 7 9 10 5 8 4 2
  88.   2. 用11对计算结果取模
  89.   Y = mod(S, 11)
  90.   3. 根据模的值得到对应的校验码
  91.   对应关系为:
  92.    Y值: 0 1 2 3 4 5 6 7 8 9 10
  93.   校验码: 1 0 X 9 8 7 6 5 4 3 2
  94. */
  95. private static boolean isVarifyCode(String Ai, String IDStr) {
  96. String[] VarifyCode = {"1", "0", "X", "9", "8", "7", "6", "5", "4", "3", "2"};
  97. String[] Wi = {"7", "9", "10", "5", "8", "4", "2", "1", "6", "3", "7", "9", "10", "5", "8", "4", "2"};
  98. int sum = 0;
  99. for (int i = 0; i < 17; i++) {
  100. sum = sum + Integer.parseInt(String.valueOf(Ai.charAt(i))) * Integer.parseInt(Wi[i]);
  101. }
  102. int modValue = sum % 11;
  103. String strVerifyCode = VarifyCode[modValue];
  104. Ai = Ai + strVerifyCode;
  105. if (IDStr.length() == 18) {
  106. if (!Ai.equals(IDStr)) {
  107. return false;
  108. }
  109. }
  110. return true;
  111. }
  112. /**
  113. * 将所有地址编码保存在一个Hashtable中
  114. *
  115. * @return Hashtable 对象
  116. */
  117. private static Hashtable<String, String> GetAreaCode() {
  118. Hashtable<String, String> hashTable = new Hashtable<>();
  119. hashTable.put("11", "北京");
  120. hashTable.put("12", "天津");
  121. hashTable.put("13", "河北");
  122. hashTable.put("14", "山西");
  123. hashTable.put("15", "内蒙古");
  124. hashTable.put("21", "辽宁");
  125. hashTable.put("22", "吉林");
  126. hashTable.put("23", "黑龙江");
  127. hashTable.put("31", "上海");
  128. hashTable.put("32", "江苏");
  129. hashTable.put("33", "浙江");
  130. hashTable.put("34", "安徽");
  131. hashTable.put("35", "福建");
  132. hashTable.put("36", "江西");
  133. hashTable.put("37", "山东");
  134. hashTable.put("41", "河南");
  135. hashTable.put("42", "湖北");
  136. hashTable.put("43", "湖南");
  137. hashTable.put("44", "广东");
  138. hashTable.put("45", "广西");
  139. hashTable.put("46", "海南");
  140. hashTable.put("50", "重庆");
  141. hashTable.put("51", "四川");
  142. hashTable.put("52", "贵州");
  143. hashTable.put("53", "云南");
  144. hashTable.put("54", "西藏");
  145. hashTable.put("61", "陕西");
  146. hashTable.put("62", "甘肃");
  147. hashTable.put("63", "青海");
  148. hashTable.put("64", "宁夏");
  149. hashTable.put("65", "新疆");
  150. hashTable.put("71", "台湾");
  151. hashTable.put("81", "香港");
  152. hashTable.put("82", "澳门");
  153. hashTable.put("91", "国外");
  154. return hashTable;
  155. }
  156. /**
  157. * 判断字符串是否为数字,0-9重复0次或者多次
  158. *
  159. */
  160. private static boolean isNumeric(String strum) {
  161. Pattern pattern = Pattern.compile("[0-9]*");
  162. Matcher isNum = pattern.matcher(strum);
  163. return isNum.matches();
  164. }
  165. /**
  166. * 功能:判断字符串出生日期是否符合正则表达式:包括年月日,闰年、平年和每月31天、30天和闰月的28天或者29天
  167. *
  168. */
  169. private static boolean isDate(String strDate) {
  170. Pattern pattern = Pattern
  171. .compile("^((\\d{2}(([02468][048])|([13579][26]))[\\-\\/\\s]?((((0?[13578])|(1[02]))[\\-\\/\\s]?((0?[1-9])|([1-2][0-9])|(3[01])))|(((0?[469])|(11))[\\-\\/\\s]?((0?[1-9])|([1-2][0-9])|(30)))|(0?2[\\-\\/\\s]?((0?[1-9])|([1-2][0-9])))))|(\\d{2}(([02468][1235679])|([13579][01345789]))[\\-\\/\\s]?((((0?[13578])|(1[02]))[\\-\\/\\s]?((0?[1-9])|([1-2][0-9])|(3[01])))|(((0?[469])|(11))[\\-\\/\\s]?((0?[1-9])|([1-2][0-9])|(30)))|(0?2[\\-\\/\\s]?((0?[1-9])|(1[0-9])|(2[0-8]))))))?$");
  172. Matcher m = pattern.matcher(strDate);
  173. return m.matches();
  174. }
  175. }
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值