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