Java身份证验证方法

复制代码
View Code
  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.Scanner;
7 import java.util.regex.Matcher;
8 import java.util.regex.Pattern;
9
10 public class IDCard {
11 /*********************************** 身份证验证开始 ****************************************/
12 /**
13 * 身份证号码验证 1、号码的结构 公民身份号码是特征组合码,由十七位数字本体码和一位校验码组成。排列顺序从左至右依次为:六位数字地址码,
14 * 八位数字出生日期码,三位数字顺序码和一位数字校验码。 2、地址码(前六位数)
15 * 表示编码对象常住户口所在县(市、旗、区)的行政区划代码,按GB/T2260的规定执行。 3、出生日期码(第七位至十四位)
16 * 表示编码对象出生的年、月、日,按GB/T7408的规定执行,年、月、日代码之间不用分隔符。 4、顺序码(第十五位至十七位)
17 * 表示在同一地址码所标识的区域范围内,对同年、同月、同日出生的人编定的顺序号, 顺序码的奇数分配给男性,偶数分配给女性。 5、校验码(第十八位数)
18 * (1)十七位数字本体码加权求和公式 S = Sum(Ai * Wi), i = 0, ... , 16 ,先对前17位数字的权求和
19 * Ai:表示第i位置上的身份证号码数字值 Wi:表示第i位置上的加权因子 Wi: 7 9 10 5 8 4 2 1 6 3 7 9 10 5 8 4
20 * 2 (2)计算模 Y = mod(S, 11) (3)通过模得到对应的校验码 Y: 0 1 2 3 4 5 6 7 8 9 10 校验码: 1 0
21 * X 9 8 7 6 5 4 3 2
22 */
23
24 /**
25 * 功能:身份证的有效验证
26 *
27 * @param IDStr
28 * 身份证号
29 * @return 有效:返回"" 无效:返回String信息
30 */
31 public static String IDCardValidate(String IDStr) {
32 String errorInfo = "";// 记录错误信息
33 String[] ValCodeArr = { "1", "0", "x", "9", "8", "7", "6", "5", "4",
34 "3", "2" };
35 String[] Wi = { "7", "9", "10", "5", "8", "4", "2", "1", "6", "3", "7",
36 "9", "10", "5", "8", "4", "2" };
37 String Ai = "";
38 // ================ 号码的长度 15位或18位 ================
39 if (IDStr.length() != 15 && IDStr.length() != 18) {
40 errorInfo = "身份证号码长度应该为15位或18位。";
41 return errorInfo;
42 }
43 // =======================(end)========================
44
45 // ================ 数字 除最后以为都为数字 ================
46 if (IDStr.length() == 18) {
47 Ai = IDStr.substring(0, 17);
48 } else if (IDStr.length() == 15) {
49 Ai = IDStr.substring(0, 6) + "19" + IDStr.substring(6, 15);
50 }
51 if (isNumeric(Ai) == false) {
52 errorInfo = "身份证15位号码都应为数字 ; 18位号码除最后一位外,都应为数字。";
53 return errorInfo;
54 }
55 // =======================(end)========================
56
57 // ================ 出生年月是否有效 ================
58 String strYear = Ai.substring(6, 10);// 年份
59 String strMonth = Ai.substring(10, 12);// 月份
60 String strDay = Ai.substring(12, 14);// 月份
61 if (isDate(strYear + "-" + strMonth + "-" + strDay) == false) {
62 errorInfo = "身份证生日无效。";
63 return errorInfo;
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 errorInfo = "身份证生日不在有效范围。";
72 return errorInfo;
73 }
74 } catch (NumberFormatException e) {
75 e.printStackTrace();
76 } catch (java.text.ParseException e) {
77 e.printStackTrace();
78 }
79 if (Integer.parseInt(strMonth) > 12 || Integer.parseInt(strMonth) == 0) {
80 errorInfo = "身份证月份无效";
81 return errorInfo;
82 }
83 if (Integer.parseInt(strDay) > 31 || Integer.parseInt(strDay) == 0) {
84 errorInfo = "身份证日期无效";
85 return errorInfo;
86 }
87 // =====================(end)=====================
88
89 // ================ 地区码时候有效 ================
90 Hashtable<?, ?> h = GetAreaCode();
91 if (h.get(Ai.substring(0, 2)) == null) {
92 errorInfo = "身份证地区编码错误。";
93 return errorInfo;
94 }
95 // ==============================================
96
97 // ================ 判断最后一位的值 ================
98 int TotalmulAiWi = 0;
99 for (int i = 0; i < 17; i++) {
100 TotalmulAiWi = TotalmulAiWi
101 + Integer.parseInt(String.valueOf(Ai.charAt(i)))
102 * Integer.parseInt(Wi[i]);
103 }
104 int modValue = TotalmulAiWi % 11;
105 String strVerifyCode = ValCodeArr[modValue];
106 Ai = Ai + strVerifyCode;
107
108 if (IDStr.length() == 18) {
109 if (Ai.equals(IDStr) == false) {
110 errorInfo = "身份证无效,不是合法的身份证号码";
111 return errorInfo;
112 }
113 } else {
114 return "YES";
115 }
116 // =====================(end)=====================
117 return "YES";
118 }
119
120 /**
121 * 功能:设置地区编码
122 *
123 * @return Hashtable 对象
124 */
125 private static Hashtable<String, String> GetAreaCode() {
126 Hashtable<String, String> hashtable = new Hashtable<String, String>();
127 hashtable.put("11", "北京");
128 hashtable.put("12", "天津");
129 hashtable.put("13", "河北");
130 hashtable.put("14", "山西");
131 hashtable.put("15", "内蒙古");
132 hashtable.put("21", "辽宁");
133 hashtable.put("22", "吉林");
134 hashtable.put("23", "黑龙江");
135 hashtable.put("31", "上海");
136 hashtable.put("32", "江苏");
137 hashtable.put("33", "浙江");
138 hashtable.put("34", "安徽");
139 hashtable.put("35", "福建");
140 hashtable.put("36", "江西");
141 hashtable.put("37", "山东");
142 hashtable.put("41", "河南");
143 hashtable.put("42", "湖北");
144 hashtable.put("43", "湖南");
145 hashtable.put("44", "广东");
146 hashtable.put("45", "广西");
147 hashtable.put("46", "海南");
148 hashtable.put("50", "重庆");
149 hashtable.put("51", "四川");
150 hashtable.put("52", "贵州");
151 hashtable.put("53", "云南");
152 hashtable.put("54", "西藏");
153 hashtable.put("61", "陕西");
154 hashtable.put("62", "甘肃");
155 hashtable.put("63", "青海");
156 hashtable.put("64", "宁夏");
157 hashtable.put("65", "新疆");
158 hashtable.put("71", "台湾");
159 hashtable.put("81", "香港");
160 hashtable.put("82", "澳门");
161 hashtable.put("91", "国外");
162 return hashtable;
163 }
164
165 /**
166 * 功能:判断字符串是否为数字
167 *
168 * @param str
169 * @return
170 */
171 private static boolean isNumeric(String str) {
172 Pattern pattern = Pattern.compile("[0-9]*");
173 Matcher isNum = pattern.matcher(str);
174 if (isNum.matches()) {
175 return true;
176 } else {
177 return false;
178 }
179 }
180
181 /**
182 * 功能:判断字符串是否为日期格式
183 *
184 * @param str
185 * @return
186 */
187 public static boolean isDate(String strDate) {
188 Pattern pattern = Pattern
189 .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])))))?{1}quot;);
190 Matcher m = pattern.matcher(strDate);
191 if (m.matches()) {
192 return true;
193 } else {
194 return false;
195 }
196 }
197
198 /**
199 * @param args
200 * @throws ParseException
201 */
202 @SuppressWarnings("static-access")
203 public static void main(String[] args) {
204 // String IDCardNum="210102820826411";
205 // String IDCardNum="210102198208264114";
206 while (true) {
207 Scanner input = new Scanner(System.in);
208 String n = input.nextLine();
209 if (n.equals("N") || n.equals("n")) {
210 break;
211 }
212 String IDCardNum = input.nextLine();
213
214 IDCard cc = new IDCard();
215 System.out.println(cc.IDCardValidate(IDCardNum));
216 }
217 // System.out.println(cc.isDate("1996-02-29"));
218 }
219 /*********************************** 身份证验证结束 ****************************************/
220
221 }
复制代码

posted @   cpcpc  阅读(5421)  评论(3编辑  收藏  举报
编辑推荐:
· 10年+ .NET Coder 心语,封装的思维:从隐藏、稳定开始理解其本质意义
· .NET Core 中如何实现缓存的预热?
· 从 HTTP 原因短语缺失研究 HTTP/2 和 HTTP/3 的设计差异
· AI与.NET技术实操系列:向量存储与相似性搜索在 .NET 中的实现
· 基于Microsoft.Extensions.AI核心库实现RAG应用
阅读排行:
· 10年+ .NET Coder 心语 ── 封装的思维:从隐藏、稳定开始理解其本质意义
· 地球OL攻略 —— 某应届生求职总结
· 提示词工程——AI应用必不可少的技术
· Open-Sora 2.0 重磅开源!
· 字符编码:从基础到乱码解决
点击右上角即可分享
微信分享提示