【Java小程序】国内身份证后台校验实现

0x01 What

身份证号码的结构

15位和18位

  • 要进行身份证号码的验证,首先要了解身份证号码的编码规则。我国身份证号码多由若干位数字或者数字与字母混合组成。
    早期身份证由15位数字构成,这主要是在1980年以前发放的身份证,后来考虑到千年虫问题, 因为15位的身份证号码只能为1900年1月1日到1999年12月31日出生的人编号,所以又增加了18位身份证号码编号规则。

18位身份证号码各位的含义:

1-2位省、自治区、直辖市代码;
3-4位地级市、盟、自治州代码;
5-6位县、县级市、区代码;
7-14位出生年月日,比如19820426代表1982年4月26日;
15-17位为顺序号,其中17位(倒数第二位)男为单数,女为双数;
18位为校验码,0-9和X。

15位身份证号码各位的含义:

1-2位省、自治区、直辖市代码;
3-4位地级市、盟、自治州代码;
5-6位县、县级市、区代码;
7-12位出生年月日,比如670401代表1967年4月1日,与18位的第一个区别;
13-15位为顺序号,其中15位男为单数,女为双数;

0x02 Why

作为尾号的校验码,是由把前十七位数字带入统一的公式计算出来的, 计算的结果是0-10,如果某人的尾号是0-9,都不会出现X,但如果尾号是10,那么就得用X来代替。 X是罗马数字的10,用X来代替10。

15位号码和18位号码的区别

多2位年份和1位识别码,把出生年月的前2位数去掉,没有最后一位的验证码,剩下就是15位身份证号码;

仅按照地域来划分

与各地的经济情况没有任何关系。

  • 1字头的为《华北区》,北京11、天津12、河北13、山西14、内蒙15
  • 2字头的为《东北区》,就是东北三省了昂。辽宁21、吉林22、黑龙江23
  • 3字头的为《华东》六省一市,上海31、江苏32、浙江33、安徽34、福建35、江西36、山东37
  • 4字头的为《华中地区+华南地区》,河南41、湖北42、湖南43、广东44、广西45、海南46
  • 5字头的为《西南地区》,重庆50、四川51、贵州52、云南53、西藏54
    • 为什么重庆的编码是50不是51,请看我的另一个回答。 (中国的身份证制度是1984年开始全国实行的。四个直辖市,其他三个都是49年建国时即设立的,但重庆是1997年才被提升为直辖市的。1984年设置身份证地区代码时,直辖市的编码一般都排在每个大区的最前面,比如华北区,北京是11,天津是12;华东区,上海是31。既然设为了直辖市,就应该与省同等级对待,重庆的身份证号码就不能再和四川一样是51了,但直辖市又得放在大区编码的最前面,所以重庆市的编码便定为了50。

所以去看重庆人家里保存的老身份证,还都是51开头的。)

  • 6字头的为《西北区》,陕西61、甘肃62、青海63、宁夏64、新疆65
  • 7字头为中华民国实际控制区域,也就是我们所说的台湾,台湾71
  • 8字头为特别行政区,香港81、澳门82
  • 9字头为海外地区,海外91

0x03 How

代码实现

package com.api.util;

import java.util.Calendar;
import java.util.GregorianCalendar;
import java.util.HashMap;
import java.util.Map;

/**
 * 身份证验证的工具(支持5位或18位省份证)
 * 身份证号码结构:
 * 17位数字和1位校验码:6位地址码数字,8位生日数字,3位出生时间顺序号,1位校验码。
 * 地址码(前6位):表示对象常住户口所在县(市、镇、区)的行政区划代码,按GB/T2260的规定执行。
 * 出生日期码,(第七位 至十四位):表示编码对象出生年、月、日,按GB按GB/T7408的规定执行,年、月、日代码之间不用分隔符。
 * 顺序码(第十五位至十七位):表示在同一地址码所标示的区域范围内,对同年、同月、同日出生的人编订的顺序号,
 * 顺序码的奇数分配给男性,偶数分配给女性。
 * 校验码(第十八位数):
 * 十七位数字本体码加权求和公式 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
 */
public class IDCardUtil {
    final static Map<Integer, String> zoneNum = new HashMap<Integer, String>();
    final static int[] PARITYBIT = {'1', '0', 'X', '9', '8', '7', '6', '5', '4', '3', '2'};
    final static int[] POWER_LIST = {7, 9, 10, 5, 8, 4, 2, 1, 6, 3, 7, 9, 10,
            5, 8, 4, 2};

    static {
        zoneNum.put(11, "北京");
        zoneNum.put(12, "天津");
        zoneNum.put(13, "河北");
        zoneNum.put(14, "山西");
        zoneNum.put(15, "内蒙古");
        zoneNum.put(21, "辽宁");
        zoneNum.put(22, "吉林");
        zoneNum.put(23, "黑龙江");
        zoneNum.put(31, "上海");
        zoneNum.put(32, "江苏");
        zoneNum.put(33, "浙江");
        zoneNum.put(34, "安徽");
        zoneNum.put(35, "福建");
        zoneNum.put(36, "江西");
        zoneNum.put(37, "山东");
        zoneNum.put(41, "河南");
        zoneNum.put(42, "湖北");
        zoneNum.put(43, "湖南");
        zoneNum.put(44, "广东");
        zoneNum.put(45, "广西");
        zoneNum.put(46, "海南");
        zoneNum.put(50, "重庆");
        zoneNum.put(51, "四川");
        zoneNum.put(52, "贵州");
        zoneNum.put(53, "云南");
        zoneNum.put(54, "西藏");
        zoneNum.put(61, "陕西");
        zoneNum.put(62, "甘肃");
        zoneNum.put(63, "青海");
        zoneNum.put(64, "新疆");
        zoneNum.put(71, "台湾");
        zoneNum.put(81, "香港");
        zoneNum.put(82, "澳门");
        zoneNum.put(91, "外国");
    }

    /**
     * 身份证验证
     *
     * @param s 号码内容
     * @return 是否有效 null和"" 都是false
     */
    public static boolean isIDCard(String certNo) {
        if (certNo == null || (certNo.length() != 15 && certNo.length() != 18))
            return false;
        final char[] cs = certNo.toUpperCase().toCharArray();
        //校验位数
        int power = 0;
        for (int i = 0; i < cs.length; i++) {
            if (i == cs.length - 1 && cs[i] == 'X')
                break;//最后一位可以 是X或x
            if (cs[i] < '0' || cs[i] > '9')
                return false;
            if (i < cs.length - 1) {
                power += (cs[i] - '0') * POWER_LIST[i];
            }
        }

        //校验区位码
        if (!zoneNum.containsKey(Integer.valueOf(certNo.substring(0, 2)))) {
            return false;
        }

        //校验年份
        String year = certNo.length() == 15 ? getIdcardCalendar() + certNo.substring(6, 8) : certNo.substring(6, 10);

        final int iyear = Integer.parseInt(year);
        if (iyear < 1900 || iyear > Calendar.getInstance().get(Calendar.YEAR))
            return false;//1900年的PASS,超过今年的PASS

        //校验月份
        String month = certNo.length() == 15 ? certNo.substring(8, 10) : certNo.substring(10, 12);
        final int imonth = Integer.parseInt(month);
        if (imonth < 1 || imonth > 12) {
            return false;
        }

        //校验天数
        String day = certNo.length() == 15 ? certNo.substring(10, 12) : certNo.substring(12, 14);
        final int iday = Integer.parseInt(day);
        if (iday < 1 || iday > 31)
            return false;

        //校验"校验码"
        if (certNo.length() == 15)
            return true;
        return cs[cs.length - 1] == PARITYBIT[power % 11];
    }

    private static int getIdcardCalendar() {
        GregorianCalendar curDay = new GregorianCalendar();
        int curYear = curDay.get(Calendar.YEAR);
        int year2bit = Integer.parseInt(String.valueOf(curYear).substring(2));
        return year2bit;
    }


    public static void main(String[] args) {
        boolean mark2 = isIDCard("53042419921021003X");
        System.out.println(mark2);
    }

}

运行结果

程序运行结果

猜你喜欢

转载自blog.csdn.net/skymyxvincent/article/details/80252407