首页 > JavaScript > js,身份证校验(正则表达式)
2009十一月20

js,身份证校验(正则表达式)

要做一个防沉迷系统,响应国家规定嘛。防沉迷呢就需要用户输入身份证号来校验一些东西的,理论上这个东西可以对付对付就行了,可是不也毕竟是个脸面,咱也总不能让年龄为200岁或3岁的小弟弟还来玩游戏吧?这样毕竟是对身体不好的。

在网上翻了翻,东抄西抄,用js写了这么前端校验,主要用的就是正则表达式,好不容易写完的,又是测试又是重构的,累啊。

做这个其实不难,在网上多找一下总会有意外收获的.但是工欲善其事,必先利其器.我们需要了解一下身份证号的规则,知己知彼百战不殆。

身份证号有15位和18位长度两种,15位的时候,都是数字,但是18位长度的时候可能有校验位(X),所以只校验数字是不够的了。

另外身份证号可以看出一个人的年龄,出生所在地(身份证所在地?)等等,这些对我们来讲都是有用的数据。

<!DOCTYPE html PUBLIC "-//W3C//DTD HTML 4.01 Transitional//EN" "http://www.w3.org/TR/html4/loose.dtd">
<html>
<head>
<meta http-equiv="Content-Type" content="text/html; charset=UTF-8">
<title>Check card no</title>
<script type='text/javascript' src='card.js'></script>
</head>
<body>
请输入身份证号码:
<input name='card_no' type='text' id='card_no' value='1234567890' />
<input type='button' name='submit' value='click me' onclick='javascript:checkCard();' />
</body>
</html>
/*
Author:yufulong
Blog:http://www.xiaoxiaozi.com
*/

var vcity={ 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:"国外"
           };

checkCard = function()
{
    var card = document.getElementById('card_no').value;
    //是否为空
    if(card === '')
    {
        alert('请输入身份证号,身份证号不能为空');
        document.getElementById('card_no').focus;
        return false;
    }
    //校验长度,类型
    if(isCardNo(card) === false)
    {
        alert('您输入的身份证号码不正确,请重新输入');
        document.getElementById('card_no').focus;
        return false;
    }
    //检查省份
    if(checkProvince(card) === false)
    {
        alert('您输入的身份证号码不正确,请重新输入');
        document.getElementById('card_no').focus;
        return false;
    }
    //校验生日
    if(checkBirthday(card) === false)
    {
        alert('您输入的身份证号码生日不正确,请重新输入');
        document.getElementById('card_no').focus();
        return false;
    }
    //检验位的检测
    if(checkParity(card) === false)
    {
        alert('您的身份证校验位不正确,请重新输入');
        document.getElementById('card_no').focus();
        return false;
    }
    alert('OK');
    return true;
};


//检查号码是否符合规范,包括长度,类型
isCardNo = function(card)
{
    //身份证号码为15位或者18位,15位时全为数字,18位前17位为数字,最后一位是校验位,可能为数字或字符X
    var reg = /(^\d{15}$)|(^\d{17}(\d|X)$)/;
    if(reg.test(card) === false)
    {
        alert("demo");
        return false;
    }

    return true;
};

//取身份证前两位,校验省份
checkProvince = function(card)
{
    var province = card.substr(0,2);
    if(vcity[province] == undefined)
    {
        return false;
    }
    return true;
};

//检查生日是否正确
checkBirthday = function(card)
{
    var len = card.length;
    //身份证15位时,次序为省(3位)市(3位)年(2位)月(2位)日(2位)校验位(3位),皆为数字
    if(len == '15')
    {
        var re_fifteen = /^(\d{6})(\d{2})(\d{2})(\d{2})(\d{3})$/;
        var arr_data = card.match(re_fifteen);
        var year = arr_data[2];
        var month = arr_data[3];
        var day = arr_data[4];
        var birthday = new Date('19'+year+'/'+month+'/'+day);
        return verifyBirthday('19'+year,month,day,birthday);
    }
    //身份证18位时,次序为省(3位)市(3位)年(4位)月(2位)日(2位)校验位(4位),校验位末尾可能为X
    if(len == '18')
    {
        var re_eighteen = /^(\d{6})(\d{4})(\d{2})(\d{2})(\d{3})([0-9]|X)$/;
        var arr_data = card.match(re_eighteen);
        var year = arr_data[2];
        var month = arr_data[3];
        var day = arr_data[4];
        var birthday = new Date(year+'/'+month+'/'+day);
        return verifyBirthday(year,month,day,birthday);
    }
    return false;
};

//校验日期
verifyBirthday = function(year,month,day,birthday)
{
    var now = new Date();
    var now_year = now.getFullYear();
    //年月日是否合理
    if(birthday.getFullYear() == year && (birthday.getMonth() + 1) == month && birthday.getDate() == day)
    {
        //判断年份的范围(3岁到100岁之间)
        var time = now_year - year;
        if(time >= 3 && time <= 100)
        {
            return true;
        }
        return false;
    }
    return false;
};

//校验位的检测
checkParity = function(card)
{
    //15位转18位
    card = changeFivteenToEighteen(card);
    var len = card.length;
    if(len == '18')
    {
        var arrInt = new Array(7, 9, 10, 5, 8, 4, 2, 1, 6, 3, 7, 9, 10, 5, 8, 4, 2);
        var arrCh = new Array('1', '0', 'X', '9', '8', '7', '6', '5', '4', '3', '2');
        var cardTemp = 0, i, valnum;
        for(i = 0; i < 17; i ++)
        {
            cardTemp += card.substr(i, 1) * arrInt[i];
        }
        valnum = arrCh[cardTemp % 11];
        if (valnum == card.substr(17, 1))
        {
            return true;
        }
        return false;
    }
    return false;
};

//15位转18位身份证号
changeFivteenToEighteen = function(card)
{
    if(card.length == '15')
    {
        var arrInt = new Array(7, 9, 10, 5, 8, 4, 2, 1, 6, 3, 7, 9, 10, 5, 8, 4, 2);
        var arrCh = new Array('1', '0', 'X', '9', '8', '7', '6', '5', '4', '3', '2');
        var cardTemp = 0, i;  
        card = card.substr(0, 6) + '19' + card.substr(6, card.length - 6);
        for(i = 0; i < 17; i ++)
        {
            cardTemp += card.substr(i, 1) * arrInt[i];
        }
        card += arrCh[cardTemp % 11];
        return card;
    }
    return card;
};

本文主要参考地址:15位和18位身份证的正则表达式及其验证

演示地址:http://www.xiaoxiaozi.com/code/js/card.html

文章作者:simaopig
本文地址:http://www.xiaoxiaozi.com/2009/11/20/1645/
版权所有 © 转载时必须以链接形式注明作者和原始出处!

12 Responses to “js,身份证校验(正则表达式)”

  1. #1 Cherry 回复 | 引用 Post:2009-11-20 17:02

    沙发

    [回复]

  2. #2 simaopig 回复 | 引用 Post:2009-11-20 17:03

    @Cherry
    晕,让你帮测试,你趁机抢沙发。。嗯。。

    [回复]

  3. #3 蜗牛仔仔 回复 | 引用 Post:2009-11-20 17:08

    好像,发出来的演示地址中的程序有问题,我用了,验证失败

    [回复]

  4. #4 Cherry 回复 | 引用 Post:2009-11-20 17:19

    @simaopig
    要的就是这个效果 哈哈

    [回复]

  5. #5 simaopig 回复 | 引用 Post:2009-11-20 17:29

    @蜗牛仔仔
    呵呵,谢谢,已经改过来了。刚才不小心改了点东西忘记了,把对象传过去了,而不是对象的值,现在好了。

    [回复]

  6. #6 LAONB 回复 | 引用 Post:2009-11-20 18:58

    我最早是用EXCEL表格提取生日,性别,校验的。

    [回复]

  7. #7 荒野无灯 回复 | 引用 Post:2009-11-20 22:06

    实用的东东,收藏之

    [回复]

  8. #8 Louis Han 回复 | 引用 Post:2009-11-21 19:56

    正则表达式绝对是正义与力量的化身啊

    [回复]

  9. #9 A.shun 回复 | 引用 Post:2009-11-22 12:27

    你的博客越来越深奥了

    [回复]

  10. #10 bolo 回复 | 引用 Post:2009-11-22 13:59

    很棒的一段代码,哈哈

    [回复]

  11. #11 飞扬轻舞 回复 | 引用 Post:2011-01-23 12:58

    不错的代码,不过还可以再完善

    15个1,校验通过

    [回复]

  12. #12 simaopig 回复 | 引用 Post:2011-01-23 16:04

    @飞扬轻舞
    15位的貌似是有问题。谢谢!

    [回复]

发表评论

:wink: :twisted: :roll: :oops: :mrgreen: :lol: :idea: :evil: :cry: :arrow: :?: :-| :-x :-o :-P :-D :-? :) :( :!: 8-O 8)