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;
        }
    }
    
}

 

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章