java實現詳細的身份證驗證、能正確驗證身份證正確性
阿新 • • 發佈:2019-01-03
package com.javazs.util; import java.text.ParseException; import java.text.SimpleDateFormat; import java.util.Calendar; import java.util.GregorianCalendar; import java.util.Hashtable; import java.util.Scanner; import java.util.regex.Matcher; import java.util.regex.Pattern; /** * 身份證號碼有效性驗證 * @author www.javazs.com * @since2016-09-28 */ public class IdCardExpUtil { /** * * @description:驗證主方法,裡面會呼叫所有方法來驗證 * @author www.javazs.com lwl * 2016年5月22日 * @param IDStr * @return * @throws ParseException */ public static String IDCardValidate(String IDStr) throws ParseException { String tipInfo = "身份證號碼正確";// 記錄錯誤資訊 String Ai = ""; // 判斷號碼的長度 15位或18位 if (IDStr.length() != 15 && IDStr.length() != 18) { tipInfo = "身份證號碼長度應該為15位或18位。"; return tipInfo; } // 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 = "身份證15位號碼都應為數字 ; 18位號碼除最後一位外,都應為數字。"; return tipInfo; } // 判斷出生年月是否有效 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 = "身份證出生日期無效。"; return tipInfo; } GregorianCalendar gc = new GregorianCalendar(); SimpleDateFormat s = new SimpleDateFormat("yyyy-MM-dd"); try { if ((gc.get(Calendar.YEAR) - Integer.parseInt(strYear)) > 150 || (gc.getTime().getTime() - s.parse( strYear + "-" + strMonth + "-" + strDay).getTime()) < 0) { tipInfo = "身份證生日不在有效範圍。"; return tipInfo; } } catch (NumberFormatException e) { e.printStackTrace(); } catch (java.text.ParseException e) { e.printStackTrace(); } if (Integer.parseInt(strMonth) > 12 || Integer.parseInt(strMonth) == 0) { tipInfo = "身份證月份無效"; return tipInfo; } if (Integer.parseInt(strDay) > 31 || Integer.parseInt(strDay) == 0) { tipInfo = "身份證日期無效"; return tipInfo; } // 判斷地區碼是否有效 Hashtable areacode = GetAreaCode(); //如果身份證前兩位的地區碼不在Hashtable,則地區碼有誤 if (areacode.get(Ai.substring(0, 2)) == null) { tipInfo = "身份證地區編碼錯誤。"; return tipInfo; } if(isVarifyCode(Ai,IDStr)==false){ tipInfo = "身份證校驗碼無效,不是合法的身份證號碼"; return tipInfo; } return tipInfo; } /* * 判斷第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 */ /** * * @description: 用於計算驗證身份證號碼最後一位的正確性 * @author www.javazs.com * 2016年5月22日 * @param Ai * @param IDStr * @return */ 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; } /** * @author www.javazs.com lwl * 將所有地址編碼儲存在一個Hashtable中 * @return Hashtable 物件 */ private static Hashtable GetAreaCode() { Hashtable hashtable = new Hashtable(); 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; } /** * @author www.javazs.com lwl * 判斷字串是否為數字,0-9重複0次或者多次 * @param strnum * @return */ private static boolean isNumeric(String strnum) { Pattern pattern = Pattern.compile("[0-9]*"); Matcher isNum = pattern.matcher(strnum); if (isNum.matches()) { return true; } else { return false; } } /** * @author www.javazs.com lwl * 功能:判斷字串出生日期是否符合正則表示式:包括年月日,閏年、平年和每月31天、30天和閏月的28天或者29天 * @param string * @return */ public static boolean isDate(String strDate) { Pattern pattern = Pattern .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]))))))?$"); Matcher m = pattern.matcher(strDate); if (m.matches()) { return true; } else { return false; } } public static void main(String[] args) throws ParseException { //String IdCard="61082120061222612X"; //從控制端輸入使用者身份證 Scanner s=new Scanner(System.in); System.out.println("請輸入你的身份證號碼:"); String IdCard=new String(s.next()); //將身份證最後一位的x轉換為大寫,便於統一 IdCard = IdCard.toUpperCase(); System.out.println(IDCardValidate(IdCard)); } }