1. 程式人生 > >java實現詳細的身份證驗證、能正確驗證身份證正確性

java實現詳細的身份證驗證、能正確驗證身份證正確性


IdCardExpUtil.java


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