1. 程式人生 > >PHP驗證身份證格式

PHP驗證身份證格式

網際網路公司對身份證驗證的需求越來越多,然而普通的小公司是無法對接公安部門的身份認證系統的。幾乎都是在網上買一些大的網際網路公司的一些認證服務。即使是便宜一些的認證價格也達到了10萬次/萬元。也就是一角錢一次了,還是挺貴哈。所以為了減少公司的開支,我們在拿到使用者提交的身份證資訊去認證前應該自己先檢驗一下身份證的格式,如果連最基本的身份證格式就不滿足的話,我們完全可以直接駁回,不必走接下來的花錢認證流程了,而且也可以建立一個使用者身份證表,把每次認證通過的資料存起來,以後每次都先從自己系統裡的使用者身份證表查詢,查不到再去走認證流程,最大化節省公司開支。
這裡提供一個初步驗證身份證格式的PHP身份證驗證類。該類只提供一個靜態方法isValid

介面,使用起來也很簡單,只需要使用IdentityCard::isValid即可驗證。可以放到自己專案中當做工具類。該方法可以同時驗證15位的老身份證和18位的新身份證格式
PHP版本: >=7.0.0
注: 如果PHP版本低於7.0版本只需要把類中方法的引數限制取消掉即可使用。

class IdentityCard
{
    /**
     * 校驗身份證號是否合法
     * @param string $num 待校驗的身份證號
     * @return bool
     */
    public static function isValid(string $num)
    {
        //老身份證長度15位,新身份證長度18位
        $length = strlen($num);
        if ($length == 15) { //如果是15位身份證

            //15位身份證沒有字母
            if (!is_numeric($num)) {
                return false;
            }
            // 省市縣(6位)
            $areaNum = substr($num, 0, 6);
            // 出生年月(6位)
            $dateNum = substr($num, 6, 6);

        } else if ($length == 18) { //如果是18位身份證

            //基本格式校驗
            if (!preg_match('/^\d{17}[0-9xX]$/', $num)) {
                return false;
            }
            // 省市縣(6位)
            $areaNum = substr($num, 0, 6);
            // 出生年月日(8位)
            $dateNum = substr($num, 6, 8);

        } else { //假身份證
            return false;
        }

        //驗證地區
        if (!self::isAreaCodeValid($areaNum)) {
            return false;
        }

        //驗證日期
        if (!self::isDateValid($dateNum)) {
            return false;
        }

        //驗證最後一位
        if (!self::isVerifyCodeValid($num)) {
            return false;
        }

        return true;
    }

    /**
     * 省市自治區校驗
     * @param string $area 省、直轄市程式碼
     * @return bool
     */
    private static function isAreaCodeValid(string $area) {
        $provinceCode = substr($area, 0, 2);

        // 根據GB/T2260—999,省市程式碼11到65
        if (11 <= $provinceCode && $provinceCode <= 65) {
            return true;
        } else {
            return false;
        }
    }

    /**
     * 驗證出生日期合法性
     * @param string $date 日期
     * @return bool
     */
    private static function isDateValid(string $date) {
        if (strlen($date) == 6) { //15位身份證號沒有年份,這裡拼上年份
            $date = '19'.$date;
        }
        $year  = intval(substr($date, 0, 4));
        $month = intval(substr($date, 4, 2));
        $day   = intval(substr($date, 6, 2));

        //日期基本格式校驗
        if (!checkdate($month, $day, $year)) {
            return false;
        }

        //日期格式正確,但是邏輯存在問題(如:年份大於當前年)
        $currYear = date('Y');
        if ($year > $currYear) {
            return false;
        }
        return true;
    }

    /**
     * 驗證18位身份證最後一位
     * @param string $num 待校驗的身份證號
     * @return bool
     */
    private static function isVerifyCodeValid(string $num)
    {
        if (strlen($num) == 18) {
            $factor = [7, 9, 10, 5, 8, 4, 2, 1, 6, 3, 7, 9, 10, 5, 8, 4, 2];
            $tokens = ['1', '0', 'X', '9', '8', '7', '6', '5', '4', '3', '2'];

            $checkSum = 0;
            for ($i = 0; $i < 17; $i++) {
                $checkSum += intval($num{$i}) * $factor[$i];
            }

            $mod   = $checkSum % 11;
            $token = $tokens[$mod];

            $lastChar = strtoupper($num{17});

            if ($lastChar != $token) {
                return false;
            }
        }
        return true;
    }
}

使用示例:

// 測試
$num = '33071919610920021X';
if (IdentityCard::isValid($num)) {
    echo '身份證格式正確';
} else {
    echo '身份證格式不正確';
}

下面是身份證編碼各個部分的含義:

//=============新的18位身份證號碼各位的含義:=======================
//1-2位省、自治區、直轄市程式碼;11-65
//3-4位地級市、盟、自治州程式碼;
//5-6位縣、縣級市、區程式碼;
//7-14位出生年月日,比如19670401代表1967年4月1日;
//15-17位為順序號,其中17位男為單數,女為雙數;
//18位為校驗碼,0-9和X,由公式隨機產生。
//舉例:
//130503 19670401 0012這個身份證號的含義: 13為河北,05為邢臺,03為橋西區,出生日期為1967年4月1日,順序號為001,2為驗證碼
//===========15位身份證號碼各位的含義:=======================
//1-2位省、自治區、直轄市程式碼;
//3-4位地級市、盟、自治州程式碼;
//5-6位縣、縣級市、區程式碼;
//7-12位出生年月日,比如670401代表1967年4月1日,這是和18位號碼的第一個區別;
//13-15位為順序號,其中15位男為單數,女為雙數;
//與18位身份證號的第二個區別:沒有最後一位的驗證碼。
//舉例:
//130503 670401 001的含義; 13為河北,05為邢臺,03為橋西區,出生日期為1967年4月1日,順序號為001。