- 工具類
package com.mazaiting;
import java.text.ParseException;
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;
/**
* 身份證號碼的格式:610821-20061222-612-X 由18位數(shù)字組成:前6位為地址碼皆看,第7至14位為出生日期碼辈讶,第15至17位為順序碼签夭,
* 第18位為校驗(yàn)碼颂砸。檢驗(yàn)碼分別是0-10共11個(gè)數(shù)字疹鳄,當(dāng)檢驗(yàn)碼為“10”時(shí),為了保證公民身份證號碼18位坟募,所以用“X”表示枯饿。雖然校驗(yàn)碼為“X”不能更換,但若需全用數(shù)字表示闹炉,只需將18位公民身份號碼轉(zhuǎn)換成15位居民身份證號碼蒿赢,去掉第7至8位和最后1位3個(gè)數(shù)碼。
* 當(dāng)今的身份證號碼有15位和18位之分渣触。1985年我國實(shí)行居民身份證制度羡棵,當(dāng)時(shí)簽發(fā)的身份證號碼是15位的,1999年簽發(fā)的身份證由于年份的擴(kuò)展(由兩位變?yōu)樗奈唬┖湍┪布恿诵?yàn)碼嗅钻,就成了18位皂冰。
* (1)前1、2位數(shù)字表示:所在省份的代碼养篓;
* (2)第3秃流、4位數(shù)字表示:所在城市的代碼;
* (3)第5觉至、6位數(shù)字表示:所在區(qū)縣的代碼剔应;
* (4)第7~14位數(shù)字表示:出生年、月语御、日峻贮;
* (5)第15、16位數(shù)字表示:所在地的派出所的代碼应闯;
* (6)第17位數(shù)字表示性別:奇數(shù)表示男性纤控,偶數(shù)表示女性
* (7)第18位數(shù)字是校檢碼:根據(jù)一定算法生成
*/
public class IdCardVerification {
/**身份證有效*/
public static final String VALIDITY = "該身份證有效!";
/**位數(shù)不足*/
public static final String LACKDIGITS = "身份證號碼長度應(yīng)該為15位或18位碉纺。";
/**最后一位應(yīng)為數(shù)字*/
public static final String LASTOFNUMBER = "身份證15位號碼都應(yīng)為數(shù)字 ; 18位號碼除最后一位外船万,都應(yīng)為數(shù)字。";
/**出生日期無效*/
public static final String INVALIDBIRTH = "身份證出生日期無效骨田。";
/**生日不在有效范圍*/
public static final String INVALIDSCOPE = "身份證生日不在有效范圍耿导。";
/**月份無效*/
public static final String INVALIDMONTH = "身份證月份無效";
/**日期無效*/
public static final String INVALIDDAY = "身份證日期無效";
/**身份證地區(qū)編碼錯(cuò)誤*/
public static final String CODINGERROR = "身份證地區(qū)編碼錯(cuò)誤。";
/**身份證校驗(yàn)碼無效*/
public static final String INVALIDCALIBRATION = "身份證校驗(yàn)碼無效态贤,不是合法的身份證號碼";
/**
* 檢驗(yàn)身份證號碼是否符合規(guī)范
* @param IDStr 身份證號碼
* @return 錯(cuò)誤信息或成功信息
*/
public static String IDCardValidate(String IDStr) throws ParseException {
String tipInfo = VALIDITY;// 記錄錯(cuò)誤信息
String Ai = "";
// 判斷號碼的長度 15位或18位
if (IDStr.length() != 15 && IDStr.length() != 18) {
tipInfo = LACKDIGITS;
return tipInfo;
}
// 18位身份證前17位位數(shù)字舱呻,如果是15位的身份證則所有號碼都為數(shù)字
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;
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 = INVALIDBIRTH;
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 = INVALIDSCOPE;
return tipInfo;
}
} catch (NumberFormatException e) {
e.printStackTrace();
} catch (java.text.ParseException e) {
e.printStackTrace();
}
if (Integer.parseInt(strMonth) > 12 || Integer.parseInt(strMonth) == 0) {
tipInfo = INVALIDMONTH;
return tipInfo;
}
if (Integer.parseInt(strDay) > 31 || Integer.parseInt(strDay) == 0) {
tipInfo = INVALIDDAY;
return tipInfo;
}
// 判斷地區(qū)碼是否有效
Hashtable<String, String> areacode = GetAreaCode();
// 如果身份證前兩位的地區(qū)碼不在Hashtable,則地區(qū)碼有誤
if (areacode.get(Ai.substring(0, 2)) == null) {
tipInfo = CODINGERROR;
return tipInfo;
}
if (isVarifyCode(Ai, IDStr) == false) {
tipInfo = INVALIDCALIBRATION;
return tipInfo;
}
return tipInfo;
}
/*
* 判斷第18位校驗(yàn)碼是否正確 第18位校驗(yàn)碼的計(jì)算方式:
* 1. 對前17位數(shù)字本體碼加權(quán)求和 公式為:S = Sum(Ai * Wi), i =
* 0, ... , 16 其中Ai表示第i個(gè)位置上的身份證號碼數(shù)字值,Wi表示第i位置上的加權(quán)因子箱吕,其各位對應(yīng)的值依次為: 7 9 10 5 8 4
* 2 1 6 3 7 9 10 5 8 4 2
* 2. 用11對計(jì)算結(jié)果取模 Y = mod(S, 11)
* 3. 根據(jù)模的值得到對應(yīng)的校驗(yàn)碼
* 對應(yīng)關(guān)系為: Y值: 0 1 2 3 4 5 6 7 8 9 10 校驗(yàn)碼: 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;
}
/**
* 將所有地址編碼保存在一個(gè)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", "內(nèi)蒙古");
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;
}
/**
* 判斷字符串是否為數(shù)字,0-9重復(fù)0次或者多次
* @param strnum
* @return true, 符合; false, 不符合芥驳。
*/
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;
}
}
/**
* 功能:判斷字符串出生日期是否符合正則表達(dá)式:包括年月日,閏年茬高、平年和每月31天兆旬、30天和閏月的28天或者29天
* @param string
* @return true, 符合; false, 不符合。
*/
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 class Test {
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轉(zhuǎn)換為大寫怎栽,便于統(tǒng)一
IdCard = IdCard.toUpperCase();
System.out.println(IdCardVerification.IDCardValidate(IdCard));
}
}