1. /*********************************** 身份證驗(yàn)證開(kāi)始 ****************************************/
2. /**
3. * 身份證號(hào)碼驗(yàn)證
4. * 1、號(hào)碼的結(jié)構(gòu)
5. * 公民身份號(hào)碼是特征組合碼,由十七位數(shù)字本體碼和一位校驗(yàn)碼組成。排列順序從左至右依次為:六位數(shù)字地址碼,
6. * 八位數(shù)字出生日期碼,三位數(shù)字順序碼和一位數(shù)字校驗(yàn)碼。
7. * 2、地址碼(前六位數(shù))
8. * 表示編碼對(duì)象常住戶口所在縣(市、旗、區(qū))的行政區(qū)劃代碼,按GB/T2260的規(guī)定執(zhí)行。
9. * 3、出生日期碼(第七位至十四位)
10. * 表示編碼對(duì)象出生的年、月、日,按GB/T7408的規(guī)定執(zhí)行,年、月、日代碼之間不用分隔符。
11. * 4、順序碼(第十五位至十七位)
12. * 表示在同一地址碼所標(biāo)識(shí)的區(qū)域范圍內(nèi),對(duì)同年、同月、同日出生的人編定的順序號(hào),
13. * 順序碼的奇數(shù)分配給男性,偶數(shù)分配給女性。
14. * 5、校驗(yàn)碼(第十八位數(shù))
15. * (1)十七位數(shù)字本體碼加權(quán)求和公式 S = Sum(Ai * Wi), i = 0,
, 16 ,先對(duì)前17位數(shù)字的權(quán)求和
16. * Ai:表示第i位置上的身份證號(hào)碼數(shù)字值 Wi:表示第i位置上的加權(quán)因子 Wi: 7 9 10 5 8 4 2 1 6 3 7 9 10 5 8 4
17. * 2 (2)計(jì)算模 Y = mod(S, 11) (3)通過(guò)模得到對(duì)應(yīng)的校驗(yàn)碼 Y: 0 1 2 3 4 5 6 7 8 9 10 校驗(yàn)碼: 1 0
18. * X 9 8 7 6 5 4 3 2
19. */
20.
21. /**
22. * 功能:身份證的有效驗(yàn)證
23. * @param IDStr 身份證號(hào)
24. * @return 有效:返回"" 無(wú)效:返回String信息
25. * @throws ParseException
26. */
27. @SuppressWarnings("unchecked")
28. public static String IDCardValidate(String IDStr) throws ParseException {
29. String errorInfo = "";// 記錄錯(cuò)誤信息
30. String[] ValCodeArr = { "1", "0", "x", "9", "8", "7", "6", "5", "4",
31. "3", "2" };
32. String[] Wi = { "7", "9", "10", "5", "8", "4", "2", "1", "6", "3", "7",
33. "9", "10", "5", "8", "4", "2" };
34. String Ai = "";
35. // ================ 號(hào)碼的長(zhǎng)度 15位或18位 ================
36. if (IDStr.length() != 15 && IDStr.length() != 18) {
37. errorInfo = "身份證號(hào)碼長(zhǎng)度應(yīng)該為15位或18位。";
38. return errorInfo;
39. }
40. // =======================(end)========================
41.
42. // ================ 數(shù)字 除最后以為都為數(shù)字 ================
43. if (IDStr.length() == 18) {
44. Ai = IDStr.substring(0, 17);
45. } else if (IDStr.length() == 15) {
46. Ai = IDStr.substring(0, 6) + "19" + IDStr.substring(6, 15);
47. }
48. if (isNumeric(Ai) == false) {
49. errorInfo = "身份證15位號(hào)碼都應(yīng)為數(shù)字 ; 18位號(hào)碼除最后一位外,都應(yīng)為數(shù)字。";
50. return errorInfo;
51. }
52. // =======================(end)========================
53.
54. // ================ 出生年月是否有效 ================
55. String strYear = Ai.substring(6, 10);// 年份
56. String strMonth = Ai.substring(10, 12);// 月份
57. String strDay = Ai.substring(12, 14);// 月份
58. if (isDate(strYear + "-" + strMonth + "-" + strDay) == false) {
59. errorInfo = "身份證生日無(wú)效。";
60. return errorInfo;
61. }
62. GregorianCalendar gc = new GregorianCalendar();
63. SimpleDateFormat s = new SimpleDateFormat("yyyy-MM-dd");
64. if ((gc.get(Calendar.YEAR) - Integer.parseInt(strYear)) > 150
65. || (gc.getTime().getTime() - s.parse(
66. strYear + "-" + strMonth + "-" + strDay).getTime()) < 0) {
67. errorInfo = "身份證生日不在有效范圍。";
68. return errorInfo;
69. }
70. if (Integer.parseInt(strMonth) > 12 || Integer.parseInt(strMonth) == 0) {
71. errorInfo = "身份證月份無(wú)效";
72. return errorInfo;
73. }
74. if (Integer.parseInt(strDay) > 31 || Integer.parseInt(strDay) == 0) {
75. errorInfo = "身份證日期無(wú)效";
76. return errorInfo;
77. }
78. // =====================(end)=====================
79.
80. // ================ 地區(qū)碼時(shí)候有效 ================
81. Hashtable h = GetAreaCode();
82. if (h.get(Ai.substring(0, 2)) == null) {
83. errorInfo = "身份證地區(qū)編碼錯(cuò)誤。";
84. return errorInfo;
85. }
86. // ==============================================
87.
88. // ================ 判斷最后一位的值 ================
89. int TotalmulAiWi = 0;
90. for (int i = 0; i < 17; i++) {
91. TotalmulAiWi = TotalmulAiWi
92. + Integer.parseInt(String.valueOf(Ai.charAt(i)))
93. * Integer.parseInt(Wi[i]);
94. }
95. int modValue = TotalmulAiWi % 11;
96. String strVerifyCode = ValCodeArr[modValue];
97. Ai = Ai + strVerifyCode;
98.
99. if (IDStr.length() == 18) {
100. if (Ai.equals(IDStr) == false) {
101. errorInfo = "身份證無(wú)效,不是合法的身份證號(hào)碼";
102. return errorInfo;
103. }
104. } else {
105. return "";
106. }
107. // =====================(end)=====================
108. return "";
109. }
110.
111. /**
112. * 功能:設(shè)置地區(qū)編碼
113. * @return Hashtable 對(duì)象
114. */
115. @SuppressWarnings("unchecked")
116. private static Hashtable GetAreaCode() {
117. Hashtable hashtable = new Hashtable();
118. hashtable.put("11", "北京");
119. hashtable.put("12", "天津");
120. hashtable.put("13", "河北");
121. hashtable.put("14", "山西");
122. hashtable.put("15", "內(nèi)蒙古");
123. hashtable.put("21", "遼寧");
124. hashtable.put("22", "吉林");
125. hashtable.put("23", "黑龍江");
126. hashtable.put("31", "上海");
127. hashtable.put("32", "江蘇");
128. hashtable.put("33", "浙江");
129. hashtable.put("34", "安徽");
130. hashtable.put("35", "福建");
131. hashtable.put("36", "江西");
132. hashtable.put("37", "山東");
133. hashtable.put("41", "河南");
134. hashtable.put("42", "湖北");
135. hashtable.put("43", "湖南");
136. hashtable.put("44", "廣東");
137. hashtable.put("45", "廣西");
138. hashtable.put("46", "海南");
139. hashtable.put("50", "重慶");
140. hashtable.put("51", "四川");
141. hashtable.put("52", "貴州");
142. hashtable.put("53", "云南");
143. hashtable.put("54", "西藏");
144. hashtable.put("61", "陜西");
145. hashtable.put("62", "甘肅");
146. hashtable.put("63", "青海");
147. hashtable.put("64", "寧夏");
148. hashtable.put("65", "新疆");
149. hashtable.put("71", "臺(tái)灣");
150. hashtable.put("81", "香港");
151. hashtable.put("82", "澳門(mén)");
152. hashtable.put("91", "國(guó)外");
153. return hashtable;
154. }
155.
156. /**
157. * 功能:判斷字符串是否為數(shù)字
158. * @param str
159. * @return
160. */
161. private static boolean isNumeric(String str) {
162. Pattern pattern = Pattern.compile("[0-9]*");
163. Matcher isNum = pattern.matcher(str);
164. if (isNum.matches()) {
165. return true;
166. } else {
167. return false;
168. }
169. }
170.
171. /**
172. * 功能:判斷字符串是否為日期格式
173. * @param str
174. * @return
175. */
176. public static boolean isDate(String strDate) {
177. Pattern pattern = Pattern
178. .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]))))))(\\s(((0?[0-9])|([1-2][0-3]))\\:([0-5]?[0-9])((\\s)|(\\:([0-5]?[0-9])))))?$");
179. Matcher m = pattern.matcher(strDate);
180. if (m.matches()) {
181. return true;
182. } else {
183. return false;
184. }
185. }
186.
187. /**
188. * @param args
189. * @throws ParseException
190. */
191. // public static void main(String[] args) throws ParseException {
192. // // String IDCardNum="210102820826411";
193. // // String IDCardNum="210102198208264114";
194. // String IDCardNum = "500113198606245216";
195. // CommonUtil cc = new CommonUtil();
196. // System.out.println(cc.IDCardValidate(IDCardNum));
197. // // System.out.println(cc.isDate("1996-02-29"));
198. // }
199. /*********************************** 身份證驗(yàn)證結(jié)束 ****************************************/
2. /**
3. * 身份證號(hào)碼驗(yàn)證
4. * 1、號(hào)碼的結(jié)構(gòu)
5. * 公民身份號(hào)碼是特征組合碼,由十七位數(shù)字本體碼和一位校驗(yàn)碼組成。排列順序從左至右依次為:六位數(shù)字地址碼,
6. * 八位數(shù)字出生日期碼,三位數(shù)字順序碼和一位數(shù)字校驗(yàn)碼。
7. * 2、地址碼(前六位數(shù))
8. * 表示編碼對(duì)象常住戶口所在縣(市、旗、區(qū))的行政區(qū)劃代碼,按GB/T2260的規(guī)定執(zhí)行。
9. * 3、出生日期碼(第七位至十四位)
10. * 表示編碼對(duì)象出生的年、月、日,按GB/T7408的規(guī)定執(zhí)行,年、月、日代碼之間不用分隔符。
11. * 4、順序碼(第十五位至十七位)
12. * 表示在同一地址碼所標(biāo)識(shí)的區(qū)域范圍內(nèi),對(duì)同年、同月、同日出生的人編定的順序號(hào),
13. * 順序碼的奇數(shù)分配給男性,偶數(shù)分配給女性。
14. * 5、校驗(yàn)碼(第十八位數(shù))
15. * (1)十七位數(shù)字本體碼加權(quán)求和公式 S = Sum(Ai * Wi), i = 0,

16. * Ai:表示第i位置上的身份證號(hào)碼數(shù)字值 Wi:表示第i位置上的加權(quán)因子 Wi: 7 9 10 5 8 4 2 1 6 3 7 9 10 5 8 4
17. * 2 (2)計(jì)算模 Y = mod(S, 11) (3)通過(guò)模得到對(duì)應(yīng)的校驗(yàn)碼 Y: 0 1 2 3 4 5 6 7 8 9 10 校驗(yàn)碼: 1 0
18. * X 9 8 7 6 5 4 3 2
19. */
20.
21. /**
22. * 功能:身份證的有效驗(yàn)證
23. * @param IDStr 身份證號(hào)
24. * @return 有效:返回"" 無(wú)效:返回String信息
25. * @throws ParseException
26. */
27. @SuppressWarnings("unchecked")
28. public static String IDCardValidate(String IDStr) throws ParseException {
29. String errorInfo = "";// 記錄錯(cuò)誤信息
30. String[] ValCodeArr = { "1", "0", "x", "9", "8", "7", "6", "5", "4",
31. "3", "2" };
32. String[] Wi = { "7", "9", "10", "5", "8", "4", "2", "1", "6", "3", "7",
33. "9", "10", "5", "8", "4", "2" };
34. String Ai = "";
35. // ================ 號(hào)碼的長(zhǎng)度 15位或18位 ================
36. if (IDStr.length() != 15 && IDStr.length() != 18) {
37. errorInfo = "身份證號(hào)碼長(zhǎng)度應(yīng)該為15位或18位。";
38. return errorInfo;
39. }
40. // =======================(end)========================
41.
42. // ================ 數(shù)字 除最后以為都為數(shù)字 ================
43. if (IDStr.length() == 18) {
44. Ai = IDStr.substring(0, 17);
45. } else if (IDStr.length() == 15) {
46. Ai = IDStr.substring(0, 6) + "19" + IDStr.substring(6, 15);
47. }
48. if (isNumeric(Ai) == false) {
49. errorInfo = "身份證15位號(hào)碼都應(yīng)為數(shù)字 ; 18位號(hào)碼除最后一位外,都應(yīng)為數(shù)字。";
50. return errorInfo;
51. }
52. // =======================(end)========================
53.
54. // ================ 出生年月是否有效 ================
55. String strYear = Ai.substring(6, 10);// 年份
56. String strMonth = Ai.substring(10, 12);// 月份
57. String strDay = Ai.substring(12, 14);// 月份
58. if (isDate(strYear + "-" + strMonth + "-" + strDay) == false) {
59. errorInfo = "身份證生日無(wú)效。";
60. return errorInfo;
61. }
62. GregorianCalendar gc = new GregorianCalendar();
63. SimpleDateFormat s = new SimpleDateFormat("yyyy-MM-dd");
64. if ((gc.get(Calendar.YEAR) - Integer.parseInt(strYear)) > 150
65. || (gc.getTime().getTime() - s.parse(
66. strYear + "-" + strMonth + "-" + strDay).getTime()) < 0) {
67. errorInfo = "身份證生日不在有效范圍。";
68. return errorInfo;
69. }
70. if (Integer.parseInt(strMonth) > 12 || Integer.parseInt(strMonth) == 0) {
71. errorInfo = "身份證月份無(wú)效";
72. return errorInfo;
73. }
74. if (Integer.parseInt(strDay) > 31 || Integer.parseInt(strDay) == 0) {
75. errorInfo = "身份證日期無(wú)效";
76. return errorInfo;
77. }
78. // =====================(end)=====================
79.
80. // ================ 地區(qū)碼時(shí)候有效 ================
81. Hashtable h = GetAreaCode();
82. if (h.get(Ai.substring(0, 2)) == null) {
83. errorInfo = "身份證地區(qū)編碼錯(cuò)誤。";
84. return errorInfo;
85. }
86. // ==============================================
87.
88. // ================ 判斷最后一位的值 ================
89. int TotalmulAiWi = 0;
90. for (int i = 0; i < 17; i++) {
91. TotalmulAiWi = TotalmulAiWi
92. + Integer.parseInt(String.valueOf(Ai.charAt(i)))
93. * Integer.parseInt(Wi[i]);
94. }
95. int modValue = TotalmulAiWi % 11;
96. String strVerifyCode = ValCodeArr[modValue];
97. Ai = Ai + strVerifyCode;
98.
99. if (IDStr.length() == 18) {
100. if (Ai.equals(IDStr) == false) {
101. errorInfo = "身份證無(wú)效,不是合法的身份證號(hào)碼";
102. return errorInfo;
103. }
104. } else {
105. return "";
106. }
107. // =====================(end)=====================
108. return "";
109. }
110.
111. /**
112. * 功能:設(shè)置地區(qū)編碼
113. * @return Hashtable 對(duì)象
114. */
115. @SuppressWarnings("unchecked")
116. private static Hashtable GetAreaCode() {
117. Hashtable hashtable = new Hashtable();
118. hashtable.put("11", "北京");
119. hashtable.put("12", "天津");
120. hashtable.put("13", "河北");
121. hashtable.put("14", "山西");
122. hashtable.put("15", "內(nèi)蒙古");
123. hashtable.put("21", "遼寧");
124. hashtable.put("22", "吉林");
125. hashtable.put("23", "黑龍江");
126. hashtable.put("31", "上海");
127. hashtable.put("32", "江蘇");
128. hashtable.put("33", "浙江");
129. hashtable.put("34", "安徽");
130. hashtable.put("35", "福建");
131. hashtable.put("36", "江西");
132. hashtable.put("37", "山東");
133. hashtable.put("41", "河南");
134. hashtable.put("42", "湖北");
135. hashtable.put("43", "湖南");
136. hashtable.put("44", "廣東");
137. hashtable.put("45", "廣西");
138. hashtable.put("46", "海南");
139. hashtable.put("50", "重慶");
140. hashtable.put("51", "四川");
141. hashtable.put("52", "貴州");
142. hashtable.put("53", "云南");
143. hashtable.put("54", "西藏");
144. hashtable.put("61", "陜西");
145. hashtable.put("62", "甘肅");
146. hashtable.put("63", "青海");
147. hashtable.put("64", "寧夏");
148. hashtable.put("65", "新疆");
149. hashtable.put("71", "臺(tái)灣");
150. hashtable.put("81", "香港");
151. hashtable.put("82", "澳門(mén)");
152. hashtable.put("91", "國(guó)外");
153. return hashtable;
154. }
155.
156. /**
157. * 功能:判斷字符串是否為數(shù)字
158. * @param str
159. * @return
160. */
161. private static boolean isNumeric(String str) {
162. Pattern pattern = Pattern.compile("[0-9]*");
163. Matcher isNum = pattern.matcher(str);
164. if (isNum.matches()) {
165. return true;
166. } else {
167. return false;
168. }
169. }
170.
171. /**
172. * 功能:判斷字符串是否為日期格式
173. * @param str
174. * @return
175. */
176. public static boolean isDate(String strDate) {
177. Pattern pattern = Pattern
178. .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]))))))(\\s(((0?[0-9])|([1-2][0-3]))\\:([0-5]?[0-9])((\\s)|(\\:([0-5]?[0-9])))))?$");
179. Matcher m = pattern.matcher(strDate);
180. if (m.matches()) {
181. return true;
182. } else {
183. return false;
184. }
185. }
186.
187. /**
188. * @param args
189. * @throws ParseException
190. */
191. // public static void main(String[] args) throws ParseException {
192. // // String IDCardNum="210102820826411";
193. // // String IDCardNum="210102198208264114";
194. // String IDCardNum = "500113198606245216";
195. // CommonUtil cc = new CommonUtil();
196. // System.out.println(cc.IDCardValidate(IDCardNum));
197. // // System.out.println(cc.isDate("1996-02-29"));
198. // }
199. /*********************************** 身份證驗(yàn)證結(jié)束 ****************************************/