import java.text.SimpleDateFormat; import java.util.Date; /** * 驗證身份證號碼 身份證號碼, 可以解析身份證號碼的各個字段, * 以及驗證身份證號碼是否有效; 身份證號碼構成:6位地址編碼+8位生日+3位順序碼+1位校驗碼 * * @ClassName: CheckIdCard * @Description: TODO * @author miemie * */ public class CheckIdCard{ private final static String BIRTH_DATE_FORMAT = "yyyyMMdd"; // 身份證號碼中的出生日期的格式 private final static Date MINIMAL_BIRTH_DATE = new Date(-2209017600000L); // 身份證的最小出生日期,1900年1月1日 private final static int NEW_CARD_NUMBER_LENGTH = 18; private final static int OLD_CARD_NUMBER_LENGTH = 15; private final static char[] VERIFY_CODE = { '1', '0', 'X', '9', '8', '7', '6', '5', '4', '3', '2' }; // 18位身份證中最后一位校驗碼 private final static int[] VERIFY_CODE_WEIGHT = { 7, 9, 10, 5, 8, 4, 2, 1, 6, 3, 7, 9, 10, 5, 8, 4, 2 };// 18位身份證中,各個數字的生成校驗碼時的權值 /** * 如果是15位身份證號碼,則自動轉換為18位 * * @param cardNumber * @return */ public static boolean check(String cardNumber){ if (null != cardNumber){ cardNumber = cardNumber.trim(); if (OLD_CARD_NUMBER_LENGTH == cardNumber.length()){ cardNumber = contertToNewCardNumber(cardNumber); } return validate(cardNumber); } return false; } public static boolean validate(String cardNumber){ boolean result = true; result = result && (null != cardNumber); // 身份證號不能為空 result = result && NEW_CARD_NUMBER_LENGTH == cardNumber.length(); // 身份證號長度是18(新證) // 身份證號的前17位必須是阿拉伯數字 for (int i = 0; result && i < NEW_CARD_NUMBER_LENGTH - 1; i++){ char ch = cardNumber.charAt(i); result = result && ch >= '0' && ch <= '9'; } // 身份證號的第18位校驗正確 result = result && (calculateVerifyCode(cardNumber) == cardNumber .charAt(NEW_CARD_NUMBER_LENGTH - 1)); // 出生日期不能晚於當前時間,並且不能早於1900年 try{ Date birthDate = new SimpleDateFormat(BIRTH_DATE_FORMAT) .parse(getBirthDayPart(cardNumber)); result = result && null != birthDate; result = result && birthDate.before(new Date()); result = result && birthDate.after(MINIMAL_BIRTH_DATE); /** * 出生日期中的年、月、日必須正確,比如月份范圍是[1,12], * 日期范圍是[1,31],還需要校驗閏年、大月、小月的情況時, * 月份和日期相符合 */ String birthdayPart = getBirthDayPart(cardNumber); String realBirthdayPart = new SimpleDateFormat(BIRTH_DATE_FORMAT) .format(birthDate); result = result && (birthdayPart.equals(realBirthdayPart)); }catch(Exception e){ result = false; } return result; } private static String getBirthDayPart(String cardNumber){ return cardNumber.substring(6, 14); } /** * 校驗碼(第十八位數): * * 十七位數字本體碼加權求和公式 S = Sum(Ai * Wi), i = 0...16 ,先對前17位數字的權求和; * Ai:表示第i位置上的身份證號碼數字值 Wi:表示第i位置上的加權因子 Wi: 7 9 10 5 8 4 2 1 6 3 7 9 10 5 8 4 * 2; 計算模 Y = mod(S, 11)< 通過模得到對應的校驗碼 Y: 0 1 2 3 4 5 6 7 8 9 10 校驗碼: 1 0 X 9 * 8 7 6 5 4 3 2 * * @param cardNumber * @return */ private static char calculateVerifyCode(CharSequence cardNumber){ int sum = 0; for (int i = 0; i < NEW_CARD_NUMBER_LENGTH - 1; i++){ char ch = cardNumber.charAt(i); sum += ((int) (ch - '0')) * VERIFY_CODE_WEIGHT[i]; } return VERIFY_CODE[sum % 11]; } /** * 把15位身份證號碼轉換到18位身份證號碼<br> * 15位身份證號碼與18位身份證號碼的區別為:<br> * 1、15位身份證號碼中,"出生年份"字段是2位,轉換時需要補入"19",表示20世紀<br> * 2、15位身份證無最后一位校驗碼。18位身份證中,校驗碼根據根據前17位生成 * * @param cardNumber * @return */ private static String contertToNewCardNumber(String oldCardNumber){ StringBuilder buf = new StringBuilder(NEW_CARD_NUMBER_LENGTH); buf.append(oldCardNumber.substring(0, 6)); buf.append("19"); buf.append(oldCardNumber.substring(6)); buf.append(CheckIdCard.calculateVerifyCode(buf)); return buf.toString(); } }