PHP校验15位和18位身份证号

前言

看新公司框架源码的时候,发现了这个功能,于是搜索一番并封装了一下身份证号校验的类。

目前大家的身份证号大多是 18 位的,当然,也不排除有些老人的身份证号是 15 位的。

如果强制要求是 18 位的话,会比较好,因为 15 位的身份证号没有校验码,可以说,只要了解大概结构,随手都可以造出一系列身份证号码来。

当然,如果只是单纯的程序校验,18 位的身份证号码也可以伪造,就是需要伪造者花点心思。

最好的还是调用相关部门给的接口,进行校验。

本文所编写的身份证号码校验,只是针对相关规则下的计算,是调用接口前能做的事情。

身份证号规则

15位: 省份(2位) + 地级市(2位) + 县级市(2位) + 出生年(2位) + 出生月(2位) + 出生日(2位) + 顺序号(3位)

18位: 省份(2位) + 地级市(2位) + 县级市(2位) + 出生年(4位) + 出生月(2位) + 出生日(2位) + 顺序号(3位) + 校验位(1位)

相比之下,18位15位 多出生年 2位、校验位 1位

其中,顺序号如果是偶数,则说明是女生,顺序号是奇数,则说明是男生。

校验位的计算:

有17位数字,分别是:

7, 9, 10, 5, 8, 4, 2, 1, 6, 3, 7, 9, 10, 5, 8, 4, 2

分别用身份证的前 17 位乘以上面相应位置的数字,然后相加。

接着用相加的和对 11 取模。

用获得的值在下面 11 个字符里查找对应位置的字符,这个字符就是校验位。

'1', '0', 'X', '9', '8', '7', '6', '5', '4', '3', '2'

15位转18位:

从上述的分析中,可以知道,只要补充上年分和校验位就可以了。

一般情况下年份补充都是加上 19 就可以了。

校验类的实现

通过分析身份证号的规则,了解到,有几点是可以做的:

  • 检查身份是否正确(一般不会变化,而且省份不多)
  • 检查地级市和县级市(如果有这方面的资源,可以考虑,不过一般不建议)
  • 检查年月日
  • 检查校验码

当然,因为可能部分人用的是 15位 的身份证号,所以需要一个转换的方法,不过,这里还是建议限制需要 18位 的身份证号。

下面开始实现:

初始化:

class IDCardFilter
{
    /**
     * 身份证号码校验
     *
     * @param  string $idCard
     * @return bool
     */
    public function vaild($idCard)
    {
        // 基础的校验,校验身份证格式是否正确
        if (!$this->isCardNumber($idCard)) {
            return false;
        }

        // 将 15 位转换成 18 位
        $idCard = $this->fifteen2Eighteen($idCard);

        // 检查省是否存在
        if (!$this->checkProvince($idCard)) {
            return false;
        }

        // 检查生日是否正确
        if (!$this->checkBirthday($idCard)) {
            return false;
        }

        // 检查校验码
        return $this->checkCode($idCard);
    }
}

上面已经实现了一个校验的方法,里面调用了类里的很多方法,下面一一实现。

检测是否是身份证号码:

这一块的处理比较简单,一个正则表达式搞定了。

其中,(^\d{15}$) 用于匹配 15位 身份证号的情况;(^\d{17}(\d|X)$) 用于匹配 18位 身份证号的情况。

const REGX = '#(^\d{15}$)|(^\d{17}(\d|X)$)#';

/**
 * 检测是否是身份证号码
 *
 * @param  string $idCard
 * @return boolean
 */
public function isCardNumber($idCard)
{
    return preg_match(self::REGX, $idCard);
}

15位转18位:

逻辑不复杂,先判断是否是15位,然后判断需要添加的年份,最终生成校验码拼接返回就OK了。

/**
 * 15位转18位
 *
 * @param  string $idCard
 * @return void
 */
public function fifteen2Eighteen($idCard)
{
    if (strlen($idCard) != 15) {
        return $idCard;
    }

    // 如果身份证顺序码是996 997 998 999,这些是为百岁以上老人的特殊编码
    // $code = array_search(substr($idCard, 12, 3), [996, 997, 998, 999]) !== false ? '18' : '19';
    // 一般 19 就够了
    $code = '19';
    $idCardBase = substr($idCard, 0, 6) . $code . substr($idCard, 6, 9);
    return $idCardBase . $this->genCode($idCardBase);
}

校验码的生成:

详细计算规则见上面,这里就不做重复的阐述了。

/**
 * 生成校验码
 *
 * @param  string $idCardBase
 * @return void
 */
final protected function genCode($idCardBase)
{
    $idCardLength = strlen($idCardBase);
    if ($idCardLength != 17) {
        return false;
    }
    $factor = [7, 9, 10, 5, 8, 4, 2, 1, 6, 3, 7, 9, 10, 5, 8, 4, 2];
    $verifyNumbers = ['1', '0', 'X', '9', '8', '7', '6', '5', '4', '3', '2'];
    $sum = 0;
    for ($i = 0; $i < $idCardLength; $i++) {
        $sum += substr($idCardBase, $i, 1) * $factor[$i];
    }
    $index = $sum % 11;
    return $verifyNumbers[$index];
}

检查省份是否正确:

protected $provinces = [
    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 => "国外"
];

/**
 * 检查省份是否正确
 *
 * @param  string $idCard
 * @return void
 */
public function checkProvince($idCard)
{
    $provinceNumber = substr($idCard, 0, 2);
    return isset($this->provinces[$provinceNumber]);
}

检测生日是否正确:

这里也是用正则匹配,匹配出年月日的。

/**
 * 检测生日是否正确
 *
 * @param  string $idCard
 * @return void
 */
public function checkBirthday($idCard)
{
    $regx = '#^\d{6}(\d{4})(\d{2})(\d{2})\d{3}[0-9X]$#';
    if (!preg_match($regx, $idCard, $matches)) {
        return false;
    }
    array_shift($matches);
    list($year, $month, $day) = $matches;
    return checkdate($month, $day, $year);
}

校验码比对:

话说,15位18位 的都完全不用考虑这个方法了。

/**
 * 校验码比对
 *
 * @param  string $idCard
 * @return void
 */
public function checkCode($idCard)
{
    $idCardBase = substr($idCard, 0, 17);
    $code = $this->genCode($idCardBase);
    return $idCard == ($idCardBase . $code);
}

完整代码

传送门:IDCardFilter

最后

这个功能最多算是新颖吧,毕竟之前没有接触过。很开心代码片段里又增加了新的成员。

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 214,504评论 6 496
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,434评论 3 389
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 160,089评论 0 349
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,378评论 1 288
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,472评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,506评论 1 292
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,519评论 3 413
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,292评论 0 270
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,738评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,022评论 2 329
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,194评论 1 342
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,873评论 5 338
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,536评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,162评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,413评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,075评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,080评论 2 352

推荐阅读更多精彩内容