1. 程式人生 > >Java驗證身份證是否合法

Java驗證身份證是否合法



import java.text.SimpleDateFormat;
import java.util.Calendar;
import java.util.GregorianCalendar;
import java.util.Hashtable;
import java.util.regex.Matcher;
import java.util.regex.Pattern;
import java.text.ParseException;


/**
 * @author open
 * 使用方法:呼叫 IDCardUtil.IDCardValidate(str)驗證是否為合法的身份證.返回的是一個String,"YES" 代表合法的身份證 ,其他值代表錯誤資訊
 *
 */
public class IDCardUtil {
	/**
	 * 驗證身份證
	 * @param IDStr
	 * @return "YES" 代表合法的身份證 ,其他值代表錯誤資訊
	 * @throws ParseException
	 */
	public static String IDCardValidate(String IDStr) {
		String tipInfo = "YES";// 記錄錯誤資訊
		String Ai = "";
		
		if(null == IDStr || IDStr.trim().isEmpty())
			return "身份證號碼長度應該為15位或18位。";
		
		// 判斷號碼的長度 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
	 */
	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 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;
	}

	/**
	 * 判斷字串是否為數字,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;
		}
	}

	/**
	 * 功能:判斷字串出生日期是否符合正則表示式:包括年月日,閏年、平年和每月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;
		}
	}

}