PHP 身份证精确匹配验证

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

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

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

备注:
作为尾号的校验码,是由把前十七位数字带入统一的公式计算出来的, 计算的结果是0-10,如果某人的尾号是0-9,都不会出现X,但如果尾号是10,那么就得用X来代替。 X是罗马数字的10,用X来代替10。 15位号码和18位号码的区别,多2位年份和1位识别码,把出生年月的前2位数去掉,没有最后一位的验证码,剩下就是15位身份证号码;

3>. 这仅仅是按照地域来划分的,与各地的经济情况没有任何关系。
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

class Idcard
{
    public $aWeight = [7,9,10,5,8,4,2,1,6,3,7,9,10,5,8,4,2]; //十七位数字本体码权重

    public $aValidate = ['1','0','X','9','8','7','6','5','4','3','2']; //mod11,对应校验码字符值

    public $birthday;//出生年月

    public $sex;//性别

    public $xingzuo;//星座

    public $shuxiang;//属相

    public function __construct(){}
    /**
     * 验证出生日期 
     */
    public  function isChinaIDCardDate($iY, $iM, $iD)
    {
        $iDate =  $iY . '-' . $iM . '-' . $iD;
        $rPattern = '/^(([0-9]{2})|(19[0-9]{2})|(20[0-9]{2}))-((0[1-9]{1})|(1[012]{1}))-((0[1-9]{1})|(1[0-9]{1})|(2[0-9]{1})|3[01]{1})$/';
        if(preg_match($rPattern, $iDate, $arr)){
            $this->birthday = $iDate;
            return true;
        }
        return false;
    }
    /**
     * 根据身份证号前17位, 算出识别码 
     */
    public function getValidateCode($id)
    {
        $id17 = substr($id,0,17);
        $sum = 0;
        $len = strlen($id17);
        for ($i=0; $i<$len; $i++){
            $sum += $id17[$i] * $this->aWeight[$i];
        }
        $mode = $sum % 11;
        return $this->aValidate[$mode];
    }
    /**
     * 验证身份证号
     */
    public function isChinaIDCard($id)
    {
         if(!$this->get_shenfen($id)){
            return false;
         }
        $len = strlen($id);
        if($len == 18){
            if (!$this->isChinaIDCardDate(substr($id,6,4), substr($id,10,2), substr($id,12,2))){
                return false;
            }
            $code = $this->getValidateCode($id);
            if (strtoupper($code) == substr($id,17,1)){
                return true;
            }
            return false;
        }
        else if($len == 15)
        {
            if(!$this->isChinaIDCardDate('19'.substr($id,6,2),substr($id,8,2),substr($id,10,2))){
                return false;
            }
            if(!is_numeric($id)){
                return false;
            }
            return true;
        }
        return false;
    }
    /**
     * 根据身份证号,自动返回对应的性别
     */
    public function getChinaIDCardSex($cid)
    {  
        $sexint = (int)substr($cid,16,1);  
        return $sexint % 2 === 0 ? '女' : '男';  
    }
    /**
     * 根据身份证号,自动返回对应的星座   
     */
    public function getChinaIDCardXZ($cid)
    {   
        $bir = substr($cid,10,4);  
        $month = (int)substr($bir,0,2);  
        $day = (int)substr($bir,2);  
        $strValue = '';  
        if(($month == 1 && $day <= 21) || ($month == 2 && $day <= 19)) {  
            $strValue = "水瓶座";  
        }else if(($month == 2 && $day > 20) || ($month == 3 && $day <= 20)) {  
            $strValue = "双鱼座";  
        }else if (($month == 3 && $day > 20) || ($month == 4 && $day <= 20)) {  
            $strValue = "白羊座";  
        }else if (($month == 4 && $day > 20) || ($month == 5 && $day <= 21)) {  
            $strValue = "金牛座";  
        }else if (($month == 5 && $day > 21) || ($month == 6 && $day <= 21)) {  
            $strValue = "双子座";  
        }else if (($month == 6 && $day > 21) || ($month == 7 && $day <= 22)) {  
            $strValue = "巨蟹座";  
        }else if (($month == 7 && $day > 22) || ($month == 8 && $day <= 23)) {  
            $strValue = "狮子座";  
        }else if (($month == 8 && $day > 23) || ($month == 9 && $day <= 23)) {  
            $strValue = "处女座";  
        }else if (($month == 9 && $day > 23) || ($month == 10 && $day <= 23)) {  
            $strValue = "天秤座";  
        }else if (($month == 10 && $day > 23) || ($month == 11 && $day <= 22)) {  
            $strValue = "天蝎座";  
        }else if (($month == 11 && $day > 22) || ($month == 12 && $day <= 21)) {  
            $strValue = "射手座";  
        }else if (($month == 12 && $day > 21) || ($month == 1 && $day <= 20)) {  
            $strValue = "魔羯座";  
        }    
        return $strValue;  
    }  
    /**
     * 根据身份证号,自动返回对应的生肖  
     */
    public function getChinaIDCardSX($cid)
    {
        $start = 1901;  
        $end = $end = (int)substr($cid,6,4);  
        $x = ($start - $end) % 12;  
        $value = "";  
        if($x == 1 || $x == -11){$value = "鼠";}  
        if($x == 0) {$value = "牛";}   
        if($x == 11 || $x == -1){$value = "虎";}  
        if($x == 10 || $x == -2){$value = "兔";}  
        if($x == 9 || $x == -3){$value = "龙";}  
        if($x == 8 || $x == -4){$value = "蛇";}  
        if($x == 7 || $x == -5){$value = "马";}  
        if($x == 6 || $x == -6){$value = "羊";}  
        if($x == 5 || $x == -7){$value = "猴";}  
        if($x == 4 || $x == -8){$value = "鸡";}  
        if($x == 3 || $x == -9){$value = "狗";}  
        if($x == 2 || $x == -10){$value = "猪";}  
        return $value;  
    }
    /**
     * 根据身份证号,自动返回对应的省、自治区、直辖市代
     */
    public function get_shenfen($id){
        $index = substr($id,0,2);
        $area = array(
            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 => "国外"
        );
        return $area[$index];
    }
}

作者: 阿里森林 |博客地址

本作品采用《CC 协议》,转载必须注明作者和本文链接
本帖由系统于 6年前 自动加精
《L05 电商实战》
从零开发一个电商项目,功能包括电商后台、商品 & SKU 管理、购物车、订单管理、支付宝支付、微信支付、订单退款流程、优惠券等
《L03 构架 API 服务器》
你将学到如 RESTFul 设计风格、PostMan 的使用、OAuth 流程,JWT 概念及使用 和 API 开发相关的进阶知识。
讨论数量: 20

嗯...只能算是一个lib,OO的方式应该是传入身份证号,然后提取信息就OK了

$idCard = new Idcard('身份证号');

//是否有效
$idCard->isVaild();

//读取性别
$idCard->sex();

//生日
$idCard->birthday();

//属相
$idCard->shuxiang();

//星座
$idCard->xingzuo();

//地址
$idCard->address();
6年前 评论

嗯...只能算是一个lib,OO的方式应该是传入身份证号,然后提取信息就OK了

$idCard = new Idcard('身份证号');

//是否有效
$idCard->isVaild();

//读取性别
$idCard->sex();

//生日
$idCard->birthday();

//属相
$idCard->shuxiang();

//星座
$idCard->xingzuo();

//地址
$idCard->address();
6年前 评论

收藏了

6年前 评论

本人以前写过个 身份校验包 identity-card,可以参考,你这星座功能我到时加上。

6年前 评论

@KevinYang 本类库能够覆盖大部分校验,依赖于政府公开标准数据(GB/T 2260-2007),当然都过去这么久,行政区划都有所变更,如果谁有最新的标准数据(目前我没有查询到GB/T 2260-2007 后续或升级标准),可以提供给我,以保证尽量做到最大化匹配。

6年前 评论

15位1是不是可以绕过

4年前 评论

港澳台的身份证号码就不能用这个了。

6年前 评论

建议写成composer 包

6年前 评论

身份证上的生日是阴历的星座就不对了

6年前 评论

判断15位身份证号那全是bug。

file
多了两个StrNo不知道怎么来的。
判断体没有return true,导致即使是正确的15位身份证号也会return false。

6年前 评论

@沙渺 @KevinYang 硬编码到code中,是为了简单且加快执行速度。当然,最近我也花了些时间写了个脚本采集1980年以来历史沿革行政区划代码,采集到的数据为纯文本文件可供其它语言调用。

传送门:crawler

打算使用 sqlite 数据库存储最终数据,等完成之后提交合并至 master 分支。

6年前 评论
沙渺

@KevinYang 地区码确实是会变的。行政区划不推荐随便的hard-code在代码里,而推荐拆分成独立的文件,从而便于靠软件包版本升级来进行变更。

6年前 评论

内容不错,身份证内容解析也很实用。不过排版稍微有点问题,建议楼主更新下

6年前 评论

@ycrao 你好,请问地区码会变吗?如果变了的话怎么办?

6年前 评论

:+1:

6年前 评论

感谢分享?

6年前 评论

手动点赞

6年前 评论

验证身份证的时候最好先应该验证一下地区是否正确public function isChinaIDCard()
if(!array_key_exists(intval(substr($idcard,0,2)),$area))
{
return false;
}

6年前 评论

干货,不错

6年前 评论

讨论应以学习和精进为目的。请勿发布不友善或者负能量的内容,与人为善,比聪明更重要!
未填写
文章
1
粉丝
13
喜欢
130
收藏
24
排名:444
访问:2.0 万
私信
所有博文
社区赞助商