android 開發中對身份證號碼的認識

開發過程中多多少少都會碰到身份證號碼,今天在做項目的時候,碰到了對對15位、18位身份證號碼進行校驗,借此機會在網上找了些這方面的資料,整理了一下,算是普及一下開發的常識。

1、第一代身份證

1.1 概念普及

第一代居民身份證是中國自為中華人民共和國公民頒發的身份證明性證件,第一階段采用印刷和照相翻拍技術塑封而成,為[聚酯薄膜]密封、單頁卡式,15位編碼。1995年7月1日起啟用新的防偽居民身份證,采用全息透視塑封套防偽。1999年10月1日起,建立和實行公民身份號碼制度,身份代碼是唯一的、終身不變的。2004年1月1日,[第二代居民身份證]開始換發,第一代居民身份證已經于2013年1月1日正式退出。

1.2 從身份證可以讀取到的信息

1)姓名

2)性別

3)民族

4)出生日期

5)常住[戶口所在地]住址(《中華人民共和國居民身份證條例》中規定的是“住址”)

6)公民身份號碼(1999年之前,居民身份證編號為15位)

7)證件的有效期(未滿十六周歲:五年;十六周歲至二十五周歲:十年;二十六周歲至四十五周歲:二十年;四十六周歲以上:長期)

8)簽發機關(第一代用印章形式,在頭像下注明XX公安局字樣)

1.3 15位碼各位的含義

1) 1-2位省、自治區、直轄市代碼;

2) 3-4位地級市、盟、自治州代碼;

3) 5-6位縣、縣級市、區代碼;

4) 7-12位出生年月日,比如670401代表1967年4月1日,與18位的第一個區別;

6) 13-15位為順序號,其中15位男為單數,女為雙數;

2、第二代身份證

2.1 概念普及

2004年3月29日起,中國大陸正式開始為居民換發內置[非接觸式IC卡]智能芯片的[第二代居民身份證],二代身份證表面采用防偽膜和[印刷防偽技術],使用個人彩色照片,并可用機器讀取數字芯片內的信息。

2.2 各位的含義

第二代居民身份證號碼共18位,從左到右的前6位數字為地址碼,緊接著的8位為出生日期,接下來的三位為順序碼,最后一位是校驗碼。總體結構框架如下。


image.png

1) 前6位是地址碼(所屬區域)。表示這個人戶口所在縣(市、區)的行政代碼。前2位代表具體省(直轄市,自治區,特別行政區),而第3、4位是城市代碼,第5、6代表區縣代碼。

2)第7-14位是出生日期,表示出生的年月日,即年(4位)+月(2位,不足2位加0)+日(2位,不足2位加0)

3)第15-17位是順序碼,表示在同一地址碼所表示的范圍內,對同年同月同日出生的人編定的順序號,順序碼的奇數分配給男性,偶數分配給女性。

4)最后1位是校驗碼,通過前17位數字根據一定的計算方法得到的,計算方式如下:


image.png
2.3 附上一份在網上找的將15位身份證號碼轉換成18位代碼并且可以校驗身份證號是否有效,忘記出處了,親測可用
public class IDCardValidateUtils {

    public static final String[] ValCodeArr = { "1", "0", "x", "9", "8", "7", "6", "5", "4", "3", "2" };
    public static final String[] Wi = { "7", "9", "10", "5", "8", "4", "2", "1", "6", "3", "7", "9", "10", "5", "8", "4", "2" };
    // 身份證的最小出生日期,1900年1月1日
    private final static Date MINIMAL_BIRTH_DATE = new Date(-2209017600000L);
    private static final String BIRTH_DATE_FORMAT="yyyyMMdd";
    private final static int NEW_CARD_NUMBER_LENGTH = 18;
    private final static int OLD_CARD_NUMBER_LENGTH = 15;
    private final static String LENGTH_ERROR="身份證長度必須為15或者18位!";
    private final static String NUMBER_ERROR="15位身份證都應該為數字,18位身份證都應該前17位應該都為數字!";
    private final static String DATE_ERROR="身份證日期驗證無效!";
    private final static String AREA_ERROR="身份證地區編碼錯誤!";
    private final static String CHECKCODE_ERROR="身份證最后一位校驗碼有誤!";
    //是否需要返回自動補全成的身份證
    private static boolean isNeedReturn_AutoCard=false;

    /**
     *
     * @param idcardNumber 需要驗證的身份證
     * @param isreturn_AutoCard 驗證無誤后,是否需要返回自動補全身份證
     * @return 身份證無誤返回傳入的身份證號
     */
    public static String validate_effective(String idcardNumber,boolean isreturn_AutoCard){
        isNeedReturn_AutoCard=isreturn_AutoCard;
        return validate_effective(idcardNumber);
    }
    /**
     * 身份證校驗
     * @param idcardNumber 需要驗證的身份證
     * @return 身份證無誤返回傳入的身份證號
     */
    public static String validate_effective(String idcardNumber){
        String Ai=idcardNumber.trim();
        System.out.println(Ai.length()!=15);
        if(Ai.length()==15|Ai.length()==18){
            //如果為15位則自動補全到18位
            if(Ai.length()==OLD_CARD_NUMBER_LENGTH){
                Ai=contertToNewCardNumber(Ai);
            }
        }else{
            return LENGTH_ERROR;
        }
        // 身份證號的前15,17位必須是阿拉伯數字
        for (int i = 0;  i < NEW_CARD_NUMBER_LENGTH - 1; i++) {
            char ch = Ai.charAt(i);
            if( ch < '0' || ch > '9'){return NUMBER_ERROR;}
        }
        //校驗身份證日期信息是否有效 ,出生日期不能晚于當前時間,并且不能早于1900年
        try {
            Date birthDate =getBirthDate(Ai);
            if(null == birthDate){
                return DATE_ERROR;
            }
            if(!birthDate.before(new Date())){
                return DATE_ERROR;
            }
            if(!birthDate.after(MINIMAL_BIRTH_DATE)){
                return DATE_ERROR;
            }
            /**
             * 出生日期中的年、月、日必須正確,比如月份范圍是[1,12],日期范圍是[1,31],還需要校驗閏年、大月、小月的情況時,
             * 月份和日期相符合
             */
            String birthdayPart = getBirthDayPart(Ai);
            String realBirthdayPart =createBirthDateParser().format(birthDate);
            if(!birthdayPart.equals(realBirthdayPart)){
                return DATE_ERROR;
            }
        } catch (Exception e) {
            return DATE_ERROR;
        }
        //校驗地區碼是否正確
        Hashtable<String, String> h = GetAreaCode();
        if (h.get(Ai.substring(0, 2)) == null) {
            return AREA_ERROR;
        }
        //校驗身份證最后一位 身份證校驗碼
        if(!calculateVerifyCode(Ai) .equals(String.valueOf(Ai.charAt(NEW_CARD_NUMBER_LENGTH - 1)))){
            return CHECKCODE_ERROR;
        }
        return isNeedReturn_AutoCard==false?idcardNumber:Ai;
    }

    /**
     * 把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(calculateVerifyCode(buf));
        return buf.toString();
    }
    /**計算最后一位校驗碼  加權值%11
     * (1)十七位數字本體碼加權求和公式 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)
     * (3)通過模得到對應的校驗碼 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 String 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')) * Integer.parseInt(Wi[i]);
        }
        return ValCodeArr[sum % 11];
    }

    /**
     * 功能:設置地區編碼
     *
     * @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;
    }

    private static Date getBirthDate(String idcard) {
        Date cacheBirthDate=null;
        try {
            cacheBirthDate = createBirthDateParser().parse(getBirthDayPart(idcard));
        } catch (Exception e) {
            throw new RuntimeException("身份證的出生日期無效");
        }
        return new Date(cacheBirthDate.getTime());
    }

    private static SimpleDateFormat createBirthDateParser() {
        return new SimpleDateFormat(BIRTH_DATE_FORMAT);
    }

    private static String getBirthDayPart(String idcardnumber) {
        return idcardnumber.substring(6, 14);
    }

}

3、澳門身份證

3.1 知識普及

澳門居民身份證(葡萄牙語:Bilhete de Identidade de Residente Macau,簡稱BIR),是澳門居民的主要身份證明文件。證件由澳門特別行政區身份證明局簽發,種類分為“澳門特別行政區永久性居民身份證”(Bilhete de Identidade de Residente Permanente da R.A.E.M.)和“澳門特別行政區非永久性居民身份證”(Bilhete de Identidade de Residente n?o Permanente da R.A.E.M.)。前者有居留權,后者沒有居留權。

每張澳門身份證均有持證人的姓名、出生日期等個人資料,并把持證人的黑白照片和簽名印在證上。年滿18周歲者,當局會發出一張有效期為十年的身份證;而未滿18歲者,其身份證有效期則為五年;年界60歲者則其居民身份證為終身。

1992年2月起,澳門發放第一代身份證;從2002年起,澳門換發了第二代智能身份證,使用 Guilloche 設計、微型印刷、彩虹印刷、紫外光印刷、鐳射刻印、多重鐳射影像、可觸性表面紋、非對稱性加密技術等防偽技術,大大加強該證的防偽功能。

3.2 澳門身份證的照片下面,會印有葡萄牙文字母(例如ASM),它代表的意思如下:

A 持證人于澳門出生

B 持證人于香港出生

C 持證人于中國大陸、臺灣出生

D 持證人于其他國家及地區出生

N 持證人出生地不明,不知道自己在何處出生


S 持證人有出生證明文件。如無出生證明文件則會漏空。

M 持證人為男性(Masculino)

F 持證人為女性(Feminino)

3.3 組成規則

澳門身份證號碼由8個拉丁數字組成,格式為:XNNNNNN(Y)。其中:

第一位X,可能是1、5、7。絕大多數人以1字開首;以5字開首的身份證號碼代表持有或曾經持有葡萄牙國民身份證或葡萄牙給外國人身份證之人士;以7字開首代表曾經取得藍卡之人士,大多都是在1970年代至1980年代期間從中國大陸持合法證件到澳門的人士。

最后一位Y,是查核用數字,是為方便電腦處理資料及檢查號碼輸入的正確性而設。

中間6位數字,是發證當局給出的順序號。

4、 臺灣身份證

4.1 簡介

中國臺灣地區的身份證稱為“國民身份證”,號碼一共有10位,第1位是大寫的英文字母,后9位是阿拉伯數字。比如:U193683453。

臺灣居民的2010版本新式身份證顏色為粉紫色,不再以身份證件上的色彩區分性別,有21項防偽功能,身份證為橫式,最新的身份證已經取消了[職業]與[籍貫]欄,背面增加[防偽][條形碼]及役別欄,身份證全部密封。臺灣人的身份證,正面是本人的身份證明,反面是配偶的身份證明,從身份證就可看出此人婚姻狀況 身份證號碼是10碼,第1碼是英文字母,不同的縣市用不同的字母 。

4.2 地區編碼

第一位大寫的英文字母是地區編碼,代表初次登記的戶籍所在地,比如,U代碼花蓮縣。每個地區編碼還對應有一個兩位數的驗證碼,用于公式驗證。臺灣的所有地區編碼和驗證碼如下:

英文 縣市 / 地區 驗證碼(數字)

A 臺北市 10

B 臺中市 11

C 基隆市 12

D 臺南市 13

E 高雄市 14

F 臺北縣 15

G 宜蘭縣 16

H 桃園縣 17

I 嘉義市 34

J 新竹縣 18

K 苗栗縣 19

L 臺中縣 20

M 南投縣 21

N 彰化縣 22

O 新竹市 35

P 云林縣 23

Q 嘉義縣 24

R 臺南縣 25

S 高雄縣 26

T 屏東縣 27

U 花蓮縣 28

V 臺東縣 29

W 金門縣 32

X 澎湖縣 30

Y 陽明山管理局 31

Z 連江縣 33

4.3 數字編碼

首位數字代表性別,男性為1、女性為2;最后一位數字是檢驗碼;其它是電腦系統給碼。

4.4 身份證驗證

最后一位數字是檢驗碼,通過公式計算得到。計算公式如下:

通算值= 首字母對應的第一位驗證碼+ 首字母對應的第二位驗證碼 * 9 + 性別碼 * 8 + 第二位數字 * 7 + 第三位數字 * 6 + 第四位數字 * 5 + 第五位數字 * 4 + 第六位數字 * 3 + 第七位數字 * 2 + 第八位數字 * 1

最后一位數 =10- 通算值的末尾數。

例如,A234567893,A對應的驗證碼是10,最后一位數是3。通算值= 1 + 09 + 28 + 37 + 46 + 55 + 64 + 73 + 82 + 9*1 = 157,通算值的末尾數是7。則10-7=3,與最后一位數(驗證碼)相同,身份證號碼正確。反之,A234567890的最后一位是0,就不是有效字號。

4.5 一些示例

U193683453,是出生地在“花蓮縣”的“男性”臺灣身分證字號。

B142610160,是出生地在“臺中市”的“男性”臺灣身分證字號。

D257856145,是出生地在“臺南市”的“女性”臺灣身分證字號。

G244431557,是出生地在“宜蘭縣”的“女性”臺灣身分證字號。

Q155304682,是出生地在“嘉義縣”的“男性”臺灣身分證字號。

F156558462,是出生地在“臺北縣”的“男性”臺灣身分證字號。

S283602231,是出生地在“高雄縣”的“女性”臺灣身分證字號。

U267087067,是出生地在“花蓮縣”的“女性”臺灣身分證字號。

5、香港身份證

5.1 知識背景

香港身份證由中華人民共和國[香港特別行政區政府]入境事務處簽發,是香港地區居民的主要身份證明文件。

[香港法例]第177章《人事登記條例》規定,凡是年滿11歲或在香港逗留多于180天人士,必須于年滿11歲后或抵港30天內登記領取身份證。每張香港身份證均有持證人的姓名、出生日期等個人資料,并把持證人的黑白照片印在證上。在姓名一欄下面,也印有中文電碼,以方便政府或機構作輸入姓名之用。目前,香港正在使用的身份證版式為2003版

5.2 種類劃分

1) 永久性居民身份證

2)居民身份證

3)智能身份證

5.3 身份證號碼

身份證號碼是:C668668(9)。

香港身份證號碼由三部分組成:一個英文字母;6個數字;括號及0-9中的任一個數字,或者字母A。括號中的數字或字母A,是校驗碼,用于檢驗括號前面的號碼的邏輯正確性。

邏輯關系:

先把首位字母改為數字,即A為1,B為2,C為3...Z為26,再乘以8;然后把字母后面的6個數字依次乘以7、6、5、4、3、2;再將以上所有乘積相加的和,除以11,得到余數;如果整除,則括號中的校驗碼為0,如果余數為1,則校驗碼為A,如果余數為2~10,則用11減去這個余數的差作校驗碼。

例如:P103265(1),P,在字母表中排行16,則以16代表,則計算校驗碼:

16×8+1×7+0×6+3×5+2×4+6×3+5×2=186

186÷11=16......余10

11-10=1,即校驗碼為1。

6 最后分享一下公司大神封裝的工具類(說實話,我沒用過,不知道過時沒,不過沒關系,了解了上面的知識,可以自己動手在這基礎上修改,不過還是可以看看里面對15位、18位、香港澳門臺灣10位身份證號碼的封裝的思路)

/**
 * 身份證工具類
 */
public class IdcardUtils {
 
    /** 中國公民身份證號碼最小長度。 */
    public static final int CHINA_ID_MIN_LENGTH = 15;
 
    /** 中國公民身份證號碼最大長度。 */
    public static final int CHINA_ID_MAX_LENGTH = 18;
 
    /** 省、直轄市代碼表 */
    public static final String cityCode[] = {
            "11", "12", "13", "14", "15", "21", "22", "23", "31", "32", "33", "34", "35", "36", "37", "41",
            "42", "43", "44", "45", "46", "50", "51", "52", "53", "54", "61", "62", "63", "64", "65", "71",
            "81", "82", "91"
    };
 
    /** 每位加權因子 */
    public static final int power[] = {
            7, 9, 10, 5, 8, 4, 2, 1, 6, 3, 7, 9, 10, 5, 8, 4, 2
    };
 
    /** 第18位校檢碼 */
    public static final String verifyCode[] = {
            "1", "0", "X", "9", "8", "7", "6", "5", "4", "3", "2"
    };
    /** 最低年限 */
    public static final int MIN = 1930;
    public static Map<String, String> cityCodes = new HashMap<String, String>();
    /** 臺灣身份首字母對應數字 */
    public static Map<String, Integer> twFirstCode = new HashMap<String, Integer>();
    /** 香港身份首字母對應數字 */
    public static Map<String, Integer> hkFirstCode = new HashMap<String, Integer>();
    static {
        cityCodes.put("11", "北京");
        cityCodes.put("12", "天津");
        cityCodes.put("13", "河北");
        cityCodes.put("14", "山西");
        cityCodes.put("15", "內蒙古");
        cityCodes.put("21", "遼寧");
        cityCodes.put("22", "吉林");
        cityCodes.put("23", "黑龍江");
        cityCodes.put("31", "上海");
        cityCodes.put("32", "江蘇");
        cityCodes.put("33", "浙江");
        cityCodes.put("34", "安徽");
        cityCodes.put("35", "福建");
        cityCodes.put("36", "江西");
        cityCodes.put("37", "山東");
        cityCodes.put("41", "河南");
        cityCodes.put("42", "湖北");
        cityCodes.put("43", "湖南");
        cityCodes.put("44", "廣東");
        cityCodes.put("45", "廣西");
        cityCodes.put("46", "海南");
        cityCodes.put("50", "重慶");
        cityCodes.put("51", "四川");
        cityCodes.put("52", "貴州");
        cityCodes.put("53", "云南");
        cityCodes.put("54", "西藏");
        cityCodes.put("61", "陜西");
        cityCodes.put("62", "甘肅");
        cityCodes.put("63", "青海");
        cityCodes.put("64", "寧夏");
        cityCodes.put("65", "新疆");
        cityCodes.put("71", "臺灣");
        cityCodes.put("81", "香港");
        cityCodes.put("82", "澳門");
        cityCodes.put("91", "國外");
        twFirstCode.put("A", 10);
        twFirstCode.put("B", 11);
        twFirstCode.put("C", 12);
        twFirstCode.put("D", 13);
        twFirstCode.put("E", 14);
        twFirstCode.put("F", 15);
        twFirstCode.put("G", 16);
        twFirstCode.put("H", 17);
        twFirstCode.put("J", 18);
        twFirstCode.put("K", 19);
        twFirstCode.put("L", 20);
        twFirstCode.put("M", 21);
        twFirstCode.put("N", 22);
        twFirstCode.put("P", 23);
        twFirstCode.put("Q", 24);
        twFirstCode.put("R", 25);
        twFirstCode.put("S", 26);
        twFirstCode.put("T", 27);
        twFirstCode.put("U", 28);
        twFirstCode.put("V", 29);
        twFirstCode.put("X", 30);
        twFirstCode.put("Y", 31);
        twFirstCode.put("W", 32);
        twFirstCode.put("Z", 33);
        twFirstCode.put("I", 34);
        twFirstCode.put("O", 35);
        hkFirstCode.put("A", 1);
        hkFirstCode.put("B", 2);
        hkFirstCode.put("C", 3);
        hkFirstCode.put("R", 18);
        hkFirstCode.put("U", 21);
        hkFirstCode.put("Z", 26);
        hkFirstCode.put("X", 24);
        hkFirstCode.put("W", 23);
        hkFirstCode.put("O", 15);
        hkFirstCode.put("N", 14);
    }
 
    /**
     * 將15位身份證號碼轉換為18位
     * 
     * @param idCard
     *            15位身份編碼
     * @return 18位身份編碼
     */
    public static String conver15CardTo18(String idCard) {
        String idCard18 = "";
        if (idCard.length() != CHINA_ID_MIN_LENGTH) {
            return null;
        }
        if (isNum(idCard)) {
            // 獲取出生年月日
            String birthday = idCard.substring(6, 12);
            Date birthDate = null;
            try {
                birthDate = new SimpleDateFormat("yyMMdd").parse(birthday);
            } catch (ParseException e) {
                e.printStackTrace();
            }
            Calendar cal = Calendar.getInstance();
            if (birthDate != null)
                cal.setTime(birthDate);
            // 獲取出生年(完全表現形式,如:2010)
            String sYear = String.valueOf(cal.get(Calendar.YEAR));
            idCard18 = idCard.substring(0, 6) + sYear + idCard.substring(8);
            // 轉換字符數組
            char[] cArr = idCard18.toCharArray();
            if (cArr != null) {
                int[] iCard = converCharToInt(cArr);
                int iSum17 = getPowerSum(iCard);
                // 獲取校驗位
                String sVal = getCheckCode18(iSum17);
                if (sVal.length() > 0) {
                    idCard18 += sVal;
                } else {
                    return null;
                }
            }
        } else {
            return null;
        }
        return idCard18;
    }
 
    /**
     * 驗證身份證是否合法
     */
    public static boolean validateCard(String idCard) {
        Log.v("IdcardUtils", "idCard = " + idCard);
        String card = idCard.trim();
        if (validateIdCard18(card)) {
            return true;
        }
        if (validateIdCard15(card)) {
            return true;
        }
        String[] cardval = validateIdCard10(card);
        if (cardval != null) {
            if (cardval[2].equals("true")) {
                return true;
            }
        }
        return false;
    }
 
    /**
     * 驗證18位身份編碼是否合法
     * 
     * @param idCard 身份編碼
     * @return 是否合法
     */
    public static boolean validateIdCard18(String idCard) {
        boolean bTrue = false;
        if (idCard.length() == CHINA_ID_MAX_LENGTH) {
            // 前17位
            String code17 = idCard.substring(0, 17);
            // 第18位
            String code18 = idCard.substring(17, CHINA_ID_MAX_LENGTH);
            if (isNum(code17)) {
                char[] cArr = code17.toCharArray();
                if (cArr != null) {
                    int[] iCard = converCharToInt(cArr);
                    int iSum17 = getPowerSum(iCard);
                    // 獲取校驗位
                    String val = getCheckCode18(iSum17);
                    if (val.length() > 0) {
                        if (val.equalsIgnoreCase(code18)) {
                            bTrue = true;
                        }
                    }
                }
            }
        }
        return bTrue;
    }
 
    /**
     * 驗證15位身份編碼是否合法
     * 
     * @param idCard
     *            身份編碼
     * @return 是否合法
     */
    public static boolean validateIdCard15(String idCard) {
        if (idCard.length() != CHINA_ID_MIN_LENGTH) {
            return false;
        }
        if (isNum(idCard)) {
            String proCode = idCard.substring(0, 2);
            if (cityCodes.get(proCode) == null) {
                return false;
            }
            String birthCode = idCard.substring(6, 12);
            Date birthDate = null;
            try {
                birthDate = new SimpleDateFormat("yy").parse(birthCode.substring(0, 2));
            } catch (ParseException e) {
                e.printStackTrace();
            }
            Calendar cal = Calendar.getInstance();
            if (birthDate != null)
                cal.setTime(birthDate);
            if (!valiDate(cal.get(Calendar.YEAR), Integer.valueOf(birthCode.substring(2, 4)),
                    Integer.valueOf(birthCode.substring(4, 6)))) {
                return false;
            }
        } else {
            return false;
        }
        return true;
    }
 
    /**
     * 驗證10位身份編碼是否合法
     * 
     * @param idCard 身份編碼
     * @return 身份證信息數組
     *         <p>
     *         [0] - 臺灣、澳門、香港 [1] - 性別(男M,女F,未知N) [2] - 是否合法(合法true,不合法false)
     *         若不是身份證件號碼則返回null
     *         </p>
     */
    public static String[] validateIdCard10(String idCard) {
        String[] info = new String[3];
        String card = idCard.replaceAll("[\\(|\\)]", "");
        if (card.length() != 8 && card.length() != 9 && idCard.length() != 10) {
            return null;
        }
        if (idCard.matches("^[a-zA-Z][0-9]{9}$")) { // 臺灣
            info[0] = "臺灣";
            System.out.println("11111");
            String char2 = idCard.substring(1, 2);
            if (char2.equals("1")) {
                info[1] = "M";
                System.out.println("MMMMMMM");
            } else if (char2.equals("2")) {
                info[1] = "F";
                System.out.println("FFFFFFF");
            } else {
                info[1] = "N";
                info[2] = "false";
                System.out.println("NNNN");
                return info;
            }
            info[2] = validateTWCard(idCard) ? "true" : "false";
        } else if (idCard.matches("^[1|5|7][0-9]{6}\\(?[0-9A-Z]\\)?$")) { // 澳門
            info[0] = "澳門";
            info[1] = "N";
            // TODO
        } else if (idCard.matches("^[A-Z]{1,2}[0-9]{6}\\(?[0-9A]\\)?$")) { // 香港
            info[0] = "香港";
            info[1] = "N";
            info[2] = validateHKCard(idCard) ? "true" : "false";
        } else {
            return null;
        }
        return info;
    }
 
    /**
     * 驗證臺灣身份證號碼
     * 
     * @param idCard
     *            身份證號碼
     * @return 驗證碼是否符合
     */
    public static boolean validateTWCard(String idCard) {
        String start = idCard.substring(0, 1);
        String mid = idCard.substring(1, 9);
        String end = idCard.substring(9, 10);
        Integer iStart = twFirstCode.get(start);
        Integer sum = iStart / 10 + (iStart % 10) * 9;
        char[] chars = mid.toCharArray();
        Integer iflag = 8;
        for (char c : chars) {
            sum = sum + Integer.valueOf(c + "") * iflag;
            iflag--;
        }
        return (sum % 10 == 0 ? 0 : (10 - sum % 10)) == Integer.valueOf(end) ? true : false;
    }
 
    /**
     * 驗證香港身份證號碼(存在Bug,部份特殊身份證無法檢查)
     * <p>
     * 身份證前2位為英文字符,如果只出現一個英文字符則表示第一位是空格,對應數字58 前2位英文字符A-Z分別對應數字10-35
     * 最后一位校驗碼為0-9的數字加上字符"A","A"代表10
     * </p>
     * <p>
     * 將身份證號碼全部轉換為數字,分別對應乘9-1相加的總和,整除11則證件號碼有效
     * </p>
     * 
     * @param idCard 身份證號碼
     * @return 驗證碼是否符合
     */
    public static boolean validateHKCard(String idCard) {
        String card = idCard.replaceAll("[\\(|\\)]", "");
        Integer sum = 0;
        if (card.length() == 9) {
            sum = (Integer.valueOf(card.substring(0, 1).toUpperCase().toCharArray()[0]) - 55) * 9
                    + (Integer.valueOf(card.substring(1, 2).toUpperCase().toCharArray()[0]) - 55) * 8;
            card = card.substring(1, 9);
        } else {
            sum = 522 + (Integer.valueOf(card.substring(0, 1).toUpperCase().toCharArray()[0]) - 55) * 8;
        }
        String mid = card.substring(1, 7);
        String end = card.substring(7, 8);
        char[] chars = mid.toCharArray();
        Integer iflag = 7;
        for (char c : chars) {
            sum = sum + Integer.valueOf(c + "") * iflag;
            iflag--;
        }
        if (end.toUpperCase().equals("A")) {
            sum = sum + 10;
        } else {
            sum = sum + Integer.valueOf(end);
        }
        return (sum % 11 == 0) ? true : false;
    }
 
    /**
     * 將字符數組轉換成數字數組
     * 
     * @param ca
     *            字符數組
     * @return 數字數組
     */
    public static int[] converCharToInt(char[] ca) {
        int len = ca.length;
        int[] iArr = new int[len];
        try {
            for (int i = 0; i < len; i++) {
                iArr[i] = Integer.parseInt(String.valueOf(ca[i]));
            }
        } catch (NumberFormatException e) {
            e.printStackTrace();
        }
        return iArr;
    }
 
    /**
     * 將身份證的每位和對應位的加權因子相乘之后,再得到和值
     * 
     * @param iArr
     * @return 身份證編碼。
     */
    public static int getPowerSum(int[] iArr) {
        int iSum = 0;
        if (power.length == iArr.length) {
            for (int i = 0; i < iArr.length; i++) {
                for (int j = 0; j < power.length; j++) {
                    if (i == j) {
                        iSum = iSum + iArr[i] * power[j];
                    }
                }
            }
        }
        return iSum;
    }
 
    /**
     * 將power和值與11取模獲得余數進行校驗碼判斷
     * 
     * @param iSum
     * @return 校驗位
     */
    public static String getCheckCode18(int iSum) {
        String sCode = "";
        switch (iSum % 11) {
        case 10:
            sCode = "2";
            break;
        case 9:
            sCode = "3";
            break;
        case 8:
            sCode = "4";
            break;
        case 7:
            sCode = "5";
            break;
        case 6:
            sCode = "6";
            break;
        case 5:
            sCode = "7";
            break;
        case 4:
            sCode = "8";
            break;
        case 3:
            sCode = "9";
            break;
        case 2:
            sCode = "x";
            break;
        case 1:
            sCode = "0";
            break;
        case 0:
            sCode = "1";
            break;
        }
        return sCode;
    }
 
    /**
     * 根據身份編號獲取年齡
     * 
     * @param idCard
     *            身份編號
     * @return 年齡
     */
    public static int getAgeByIdCard(String idCard) {
        int iAge = 0;
        if (TextUtils.isEmpty(idCard)) {
            return 0;
        }
        if (idCard.length() == CHINA_ID_MIN_LENGTH) {
            idCard = conver15CardTo18(idCard);
        }
        String year = idCard.substring(6, 10);
        Calendar cal = Calendar.getInstance();
        int iCurrYear = cal.get(Calendar.YEAR);
        iAge = iCurrYear - Integer.valueOf(year);
        return iAge;
    }
 
    /**
     * 根據身份編號獲取生日
     * 
     * @param idCard 身份編號
     * @return 生日(yyyyMMdd)
     */
    public static String getBirthByIdCard(String idCard) {
        Integer len = idCard.length();
        if (len < CHINA_ID_MIN_LENGTH) {
            return null;
        } else if (len == CHINA_ID_MIN_LENGTH) {
            idCard = conver15CardTo18(idCard);
        }
        return idCard.substring(6, 14);
    }
 
    /**
     * 根據身份編號獲取生日年
     * 
     * @param idCard 身份編號
     * @return 生日(yyyy)
     */
    public static Short getYearByIdCard(String idCard) {
        Integer len = idCard.length();
        if (len < CHINA_ID_MIN_LENGTH) {
            return null;
        } else if (len == CHINA_ID_MIN_LENGTH) {
            idCard = conver15CardTo18(idCard);
        }
        return Short.valueOf(idCard.substring(6, 10));
    }
 
    /**
     * 根據身份編號獲取生日月
     * 
     * @param idCard
     *            身份編號
     * @return 生日(MM)
     */
    public static Short getMonthByIdCard(String idCard) {
        Integer len = idCard.length();
        if (len < CHINA_ID_MIN_LENGTH) {
            return null;
        } else if (len == CHINA_ID_MIN_LENGTH) {
            idCard = conver15CardTo18(idCard);
        }
        return Short.valueOf(idCard.substring(10, 12));
    }
 
    /**
     * 根據身份編號獲取生日天
     * 
     * @param idCard
     *            身份編號
     * @return 生日(dd)
     */
    public static Short getDateByIdCard(String idCard) {
        Integer len = idCard.length();
        if (len < CHINA_ID_MIN_LENGTH) {
            return null;
        } else if (len == CHINA_ID_MIN_LENGTH) {
            idCard = conver15CardTo18(idCard);
        }
        return Short.valueOf(idCard.substring(12, 14));
    }
 
    /**
     * 根據身份編號獲取性別
     * 
     * @param idCard 身份編號
     * @return 性別(M-男,F-女,N-未知)
     */
    public static String getGenderByIdCard(String idCard) {
        String sGender = "N";
        if (idCard.length() == CHINA_ID_MIN_LENGTH) {
            idCard = conver15CardTo18(idCard);
        }
        String sCardNum = idCard.substring(16, 17);
        if (Integer.parseInt(sCardNum) % 2 != 0) {
            sGender = "M";
        } else {
            sGender = "F";
        }
        return sGender;
    }
 
    /**
     * 根據身份編號獲取戶籍省份
     * 
     * @param idCard 身份編碼
     * @return 省級編碼。
     */
    public static String getProvinceByIdCard(String idCard) {
        int len = idCard.length();
        String sProvince = null;
        String sProvinNum = "";
        if (len == CHINA_ID_MIN_LENGTH || len == CHINA_ID_MAX_LENGTH) {
            sProvinNum = idCard.substring(0, 2);
        }
        sProvince = cityCodes.get(sProvinNum);
        return sProvince;
    }
 
    /**
     * 數字驗證
     * 
     * @param val
     * @return 提取的數字。
     */
    public static boolean isNum(String val) {
        return val == null || "".equals(val) ? false : val.matches("^[0-9]*$");
    }
 
    /**
     * 驗證小于當前日期 是否有效
     * 
     * @param iYear
     *            待驗證日期(年)
     * @param iMonth
     *            待驗證日期(月 1-12)
     * @param iDate
     *            待驗證日期(日)
     * @return 是否有效
     */
    public static boolean valiDate(int iYear, int iMonth, int iDate) {
        Calendar cal = Calendar.getInstance();
        int year = cal.get(Calendar.YEAR);
        int datePerMonth;
        if (iYear < MIN || iYear >= year) {
            return false;
        }
        if (iMonth < 1 || iMonth > 12) {
            return false;
        }
        switch (iMonth) {
        case 4:
        case 6:
        case 9:
        case 11:
            datePerMonth = 30;
            break;
        case 2:
            boolean dm = ((iYear % 4 == 0 && iYear % 100 != 0) || (iYear % 400 == 0))
                    && (iYear > MIN && iYear < year);
            datePerMonth = dm ? 29 : 28;
            break;
        default:
            datePerMonth = 31;
        }
        return (iDate >= 1) && (iDate <= datePerMonth);
    }
}

參考文章

http://shenfenzheng.bajiu.cn/?rid=39

https://baike.baidu.com/item/%E5%8F%B0%E6%B9%BE%E8%BA%AB%E4%BB%BD%E8%AF%81/6469958?fr=aladdin

http://shenfenzheng.bajiu.cn/?rid=40

https://baike.baidu.com/item/%E9%A6%99%E6%B8%AF%E8%BA%AB%E4%BB%BD%E8%AF%81/2276968?fr=aladdin

?著作權歸作者所有,轉載或內容合作請聯系作者
  • 序言:七十年代末,一起剝皮案震驚了整個濱河市,隨后出現的幾起案子,更是在濱河造成了極大的恐慌,老刑警劉巖,帶你破解...
    沈念sama閱讀 227,663評論 6 531
  • 序言:濱河連續發生了三起死亡事件,死亡現場離奇詭異,居然都是意外死亡,警方通過查閱死者的電腦和手機,發現死者居然都...
    沈念sama閱讀 98,125評論 3 414
  • 文/潘曉璐 我一進店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人,你說我怎么就攤上這事。” “怎么了?”我有些...
    開封第一講書人閱讀 175,506評論 0 373
  • 文/不壞的土叔 我叫張陵,是天一觀的道長。 經常有香客問我,道長,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 62,614評論 1 307
  • 正文 為了忘掉前任,我火速辦了婚禮,結果婚禮上,老公的妹妹穿的比我還像新娘。我一直安慰自己,他們只是感情好,可當我...
    茶點故事閱讀 71,402評論 6 404
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著,像睡著了一般。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發上,一...
    開封第一講書人閱讀 54,934評論 1 321
  • 那天,我揣著相機與錄音,去河邊找鬼。 笑死,一個胖子當著我的面吹牛,可吹牛的內容都是我干的。 我是一名探鬼主播,決...
    沈念sama閱讀 43,021評論 3 440
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了?” 一聲冷哼從身側響起,我...
    開封第一講書人閱讀 42,168評論 0 287
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎,沒想到半個月后,有當地人在樹林里發現了一具尸體,經...
    沈念sama閱讀 48,690評論 1 333
  • 正文 獨居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內容為張勛視角 年9月15日...
    茶點故事閱讀 40,596評論 3 354
  • 正文 我和宋清朗相戀三年,在試婚紗的時候發現自己被綠了。 大學時的朋友給我發了我未婚夫和他白月光在一起吃飯的照片。...
    茶點故事閱讀 42,784評論 1 369
  • 序言:一個原本活蹦亂跳的男人離奇死亡,死狀恐怖,靈堂內的尸體忽然破棺而出,到底是詐尸還是另有隱情,我是刑警寧澤,帶...
    沈念sama閱讀 38,288評論 5 357
  • 正文 年R本政府宣布,位于F島的核電站,受9級特大地震影響,放射性物質發生泄漏。R本人自食惡果不足惜,卻給世界環境...
    茶點故事閱讀 44,027評論 3 347
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧,春花似錦、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 34,404評論 0 25
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至,卻和暖如春,著一層夾襖步出監牢的瞬間,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 35,662評論 1 280
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留,地道東北人。 一個月前我還...
    沈念sama閱讀 51,398評論 3 390
  • 正文 我出身青樓,卻偏偏與公主長得像,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個殘疾皇子,可洞房花燭夜當晚...
    茶點故事閱讀 47,743評論 2 370

推薦閱讀更多精彩內容

  • 為什么有的人身份證最后一位的號碼是0、1或2、3、4、5、6、7、8、9或X,而有的人是X這是怎么回事的呢,又有多...
    吳富良閱讀 7,564評論 0 1
  • 簡言 在做用戶實名驗證時,常會用到身份證號碼的正則表達式及校驗方案。本文列舉了兩種驗證方案,大家可以根據自己的項目...
    毛三十閱讀 1,249評論 0 5
  • SeekBar是滑動條組件,在音視頻的播放器的下面經常看到。 該組件的屬性: ·android:max設置進度條的...
    風行草偃閱讀 11,446評論 0 3
  • 古語有云:知己知彼,才能百戰不殆。 在銷售工作中也是如此,作為業務人員首先要具有偵查兵一樣的嗅覺,提前做好情報搜集...
    hello貢閱讀 304評論 0 2
  • 這周由于是寒假再回家的一個過程,所以就見了很多朋友。似乎每次離開一個地方再去一個地方,都要好好的說聲再見。 在家的...
    DL諾爾閱讀 197評論 0 1