1. 程式人生 > 其它 >Java驗證身份證號碼的格式

Java驗證身份證號碼的格式

直接上程式碼

import java.text.ParseException;
import java.util.Calendar;
import java.util.GregorianCalendar;
import java.util.HashMap;
import java.util.Hashtable;
import java.util.Map;
import java.util.regex.Matcher;

/**
 * 身份證號碼的格式:610821-20061222-612-X 由18位數字組成:前6位為地址碼,第7至14位為出生日期碼,第15至17位為順序碼,
 * 第18位為校驗碼。檢驗碼分別是0-10共11個數字,當檢驗碼為“10”時,為了保證公民身份證號碼18位,所以用“X”表示。雖然校驗碼為“X”不能更換,但若需全用數字表示,只需將18位公民身份號碼轉換成15位居民身份證號碼,去掉第7至8位和最後1位3個數碼。
 * 當今的身份證號碼有15位和18位之分。1985年我國實行居民身份證制度,當時簽發的身份證號碼是15位的,1999年簽發的身份證由於年份的擴充套件(由兩位變為四位)和末尾加了效驗碼,就成了18位。
 * (1)前1、2位數字表示:所在省份的程式碼; 
 * (2)第3、4位數字表示:所在城市的程式碼; 
 * (3)第5、6位數字表示:所在區縣的程式碼;
 * (4)第7~14位數字表示:出生年、月、日; 
 * (5)第15、16位數字表示:所在地的派出所的程式碼; 
 * (6)第17位數字表示性別:奇數表示男性,偶數表示女性
 * (7)第18位數字是校檢碼:根據一定演算法生成
 
*/ public class IdCardVerification { /**身份證有效*/ public static final String VALIDITY = "該身份證有效!"; /**位數不足*/ public static final String LACKDIGITS = "身份證號碼長度應該為15位或18位。"; /**最後一位應為數字*/ public static final String LASTOFNUMBER = "身份證15位號碼都應為數字 ; 18位號碼除最後一位外,都應為數字。"; /**出生日期無效*/ public
static final String INVALIDBIRTH = "身份證出生日期無效。"; /**生日不在有效範圍*/ public static final String INVALIDSCOPE = "身份證生日不在有效範圍。"; /**月份無效*/ public static final String INVALIDMONTH = "身份證月份無效"; /**日期無效*/ public static final String INVALIDDAY = "身份證日期無效"; /**身份證地區編碼錯誤*/ public static
final String CODINGERROR = "身份證地區編碼錯誤。"; /**身份證校驗碼無效*/ public static final String INVALIDCALIBRATION = "身份證校驗碼無效,不是合法的身份證號碼"; /** * 檢驗身份證號碼是否符合規範 * @param IDStr 身份證號碼 * @return 錯誤資訊或成功資訊 */ public static Map<String,Object> IDCardValidate(String IDStr) throws ParseException { Map<String,Object> map = new HashMap<String,Object>(); String tipInfo = VALIDITY;// 記錄錯誤資訊 String Ai = ""; // 判斷號碼的長度 15位或18位 if (IDStr.length() != 15 && IDStr.length() != 18) { tipInfo = LACKDIGITS; map.put("key", false); map.put("value", tipInfo); return map; } // 18位身份證前17位位數字,如果是15位的身份證則所有號碼都為數字 if (IDStr.length() == 18) { Ai = IDStr.substring(0, 17); } else if (IDStr.length() == 15) { Ai = IDStr.substring(0, 6) + "19" + IDStr.substring(6, 15); } if (isNumeric(Ai) == false) { tipInfo = LASTOFNUMBER; map.put("key", false); map.put("value", tipInfo); return map; } // 判斷出生年月是否有效 String strYear = Ai.substring(6, 10);// 年份 String strMonth = Ai.substring(10, 12);// 月份 String strDay = Ai.substring(12, 14);// 日期 if (isDate(strYear + "-" + strMonth + "-" + strDay) == false) { tipInfo = INVALIDBIRTH; map.put("key", false); map.put("value", tipInfo); return map; } GregorianCalendar gc = new GregorianCalendar(); try { if ((gc.get(Calendar.YEAR) - Integer.parseInt(strYear)) > 150 || (gc.getTime().getTime() - Constant.SDF1.parse(strYear + "-" + strMonth + "-" + strDay).getTime()) < 0) { tipInfo = INVALIDSCOPE; map.put("key", false); map.put("value", tipInfo); return map; } } catch (NumberFormatException e) { e.printStackTrace(); } catch (java.text.ParseException e) { e.printStackTrace(); } if (Integer.parseInt(strMonth) > 12 || Integer.parseInt(strMonth) == 0) { tipInfo = INVALIDMONTH; map.put("key", false); map.put("value", tipInfo); return map; } if (Integer.parseInt(strDay) > 31 || Integer.parseInt(strDay) == 0) { tipInfo = INVALIDDAY; map.put("key", false); map.put("value", tipInfo); return map; } // 判斷地區碼是否有效 Hashtable<String, String> areacode = GetAreaCode(); // 如果身份證前兩位的地區碼不在Hashtable,則地區碼有誤 if (areacode.get(Ai.substring(0, 2)) == null) { tipInfo = CODINGERROR; map.put("key", false); map.put("value", tipInfo); return map; } if (isVarifyCode(Ai, IDStr) == false) { tipInfo = INVALIDCALIBRATION; map.put("key", false); map.put("value", tipInfo); return map; } map.put("key", true); map.put("value", tipInfo); return map; } /* * 判斷第18位校驗碼是否正確 第18位校驗碼的計算方式: * 1. 對前17位數字本體碼加權求和 公式為:S = Sum(Ai * Wi), i = * 0, ... , 16 其中Ai表示第i個位置上的身份證號碼數字值,Wi表示第i位置上的加權因子,其各位對應的值依次為: 7 9 10 5 8 4 * 2 1 6 3 7 9 10 5 8 4 2 * 2. 用11對計算結果取模 Y = mod(S, 11) * 3. 根據模的值得到對應的校驗碼 * 對應關係為: Y值: 0 1 2 3 4 5 6 7 8 9 10 校驗碼: 1 0 X 9 8 7 6 5 4 3 2 */ private static boolean isVarifyCode(String Ai, String IDStr) { String[] VarifyCode = { "1", "0", "X", "9", "8", "7", "6", "5", "4", "3", "2" }; String[] Wi = { "7", "9", "10", "5", "8", "4", "2", "1", "6", "3", "7", "9", "10", "5", "8", "4", "2" }; int sum = 0; for (int i = 0; i < 17; i++) { sum = sum + Integer.parseInt(String.valueOf(Ai.charAt(i))) * Integer.parseInt(Wi[i]); } int modValue = sum % 11; String strVerifyCode = VarifyCode[modValue]; Ai = Ai + strVerifyCode; if (IDStr.length() == 18) { if (Ai.equals(IDStr) == false) { return false; } } return true; } /** * 將所有地址編碼儲存在一個Hashtable中 * @return Hashtable 物件 */ private static Hashtable<String, String> GetAreaCode() { Hashtable<String, String> hashtable = new Hashtable<String, String>(); hashtable.put("11", "北京"); hashtable.put("12", "天津"); hashtable.put("13", "河北"); hashtable.put("14", "山西"); hashtable.put("15", "內蒙古"); hashtable.put("21", "遼寧"); hashtable.put("22", "吉林"); hashtable.put("23", "黑龍江"); hashtable.put("31", "上海"); hashtable.put("32", "江蘇"); hashtable.put("33", "浙江"); hashtable.put("34", "安徽"); hashtable.put("35", "福建"); hashtable.put("36", "江西"); hashtable.put("37", "山東"); hashtable.put("41", "河南"); hashtable.put("42", "湖北"); hashtable.put("43", "湖南"); hashtable.put("44", "廣東"); hashtable.put("45", "廣西"); hashtable.put("46", "海南"); hashtable.put("50", "重慶"); hashtable.put("51", "四川"); hashtable.put("52", "貴州"); hashtable.put("53", "雲南"); hashtable.put("54", "西藏"); hashtable.put("61", "陝西"); hashtable.put("62", "甘肅"); hashtable.put("63", "青海"); hashtable.put("64", "寧夏"); hashtable.put("65", "新疆"); hashtable.put("71", "臺灣"); hashtable.put("81", "香港"); hashtable.put("82", "澳門"); hashtable.put("91", "國外"); return hashtable; } /** * 判斷字串是否為數字,0-9重複0次或者多次 * @param strnum * @return true, 符合; false, 不符合。 */ private static boolean isNumeric(String strnum) { Matcher isNum = Constant.p2.matcher(strnum); if (isNum.matches()) { return true; } else { return false; } } /** * 功能:判斷字串出生日期是否符合正則表示式:包括年月日,閏年、平年和每月31天、30天和閏月的28天或者29天 * @param string * @return true, 符合; false, 不符合。 */ public static boolean isDate(String strDate) { Matcher m = Constant.p3.matcher(strDate); if (m.matches()) { return true; } else { return false; } } }