IMEI码规则及校验算法

注:本文章部分内容来自简书,请点击这里获取,侵删。

1. IMEI码规则

IMEI码由15-17位数字组成。

·第一部分 TAC,Type Allocation Code,类型分配码,由8位数字组成(早期是6位),是区分手机品牌和型号的编码,该代码由GSMA及其授权机构分配。其中TAC码前两位又是分配机构标识(Reporting Body Identifier),是授权IMEI码分配机构的代码,如01为美国CTIA,35为英国BABT,86为中国TAF。

·第二部分 FAC,Final Assembly Code,最终装配地代码,由2位数字构成,仅在早期TAC码为6位的手机中存在,所以TAC和FAC码合计一共8位数字。FAC码用于生产商内部区分生产地代码。

·第三部分 SNR,Serial Number,序列号,由第9位开始的6位数字组成,区分每部手机的生产序列号。

·第四部分 CD,Check Digit,验证码,由前14位数字通过Luhn算法计算得出。

·第五部分 SVN,Software Version Number,软件版本号,区分同型号手机出厂时使用的不同软件版本,仅在部分品牌的部分机型中存在。

具体组成格式如下图所示:

TACTACTACTACTACTACFACFACSNRSNRSNRSNRSNRSNRSNRCD
D15D14D13D12D11D10D9D8D7D6D5D4D3D2D1D0
3531140080963666

2. IMEI校验算法介绍及实现

2.1 算法介绍

·计算IMEI验证码的步骤:
·1、把IMEI的奇数位数*2,如:D1,D3,D5,……D13
D13 D11 D9 D7 D5 D3 D1
10 2 8 0 0 12 12

·2、将计算得到的7个奇数位数字分别以个位数相加(如果得到的是个两位数,则十位和个位分别当成个位数来相加),再加上7个偶数位数字,如:D2,D4,D6……D14
3+1+0+3+2+1+8+0+0+8+0+9+1+2+3+1+2=44
·3、如果第2步计算得到的数字末位为0,则验证码数字为0。如果第2步计算结果末位数不是0,则以大于第2步计算结果的以0结尾的双位整数减去第2步的计算结果,所获得的个位数即为验证码。
D0 = 50 -44 =6

2.2 算法实现

2.2.1 C语言实现
//返回iemi第15位校验位
char GetIMEICheckDigit(char *pp_Imei)
{
    int i;
    int vl_Sum1 = 0, vl_Sum2 = 0, vl_Total = 0;
    int vl_Temp = 0;

    for (i = 0; i<14; i++)
    {
        /*(1)将奇数位数字相加(从1开始计数)*/
        if ((i % 2) == 0)
        {
            vl_Sum1 = vl_Sum1 + pp_Imei[i] - '0';
        }
        else
        {
            /*(2)将偶数位数字分别乘以2,分别计算个位数和十位数之和(从1开始计数)*/
            vl_Temp = (pp_Imei[i] - '0') * 2;
            if (vl_Temp < 10)
            {
                vl_Sum2 = vl_Sum2 + vl_Temp;
            }
            else
            {
                vl_Sum2 = vl_Sum2 + 1 + vl_Temp - 10;
            }
        }
    }

    /*(1)+(2)*/
    vl_Total = vl_Sum1 + vl_Sum2;

    /*如果得出的数个位是0则校验位为0,否则为10减去个位数 */
    if ((vl_Total % 10) == 0)
    {
        return '0';
    }
    else
    {
        return (char)(10 - (vl_Total % 10) + '0');
    }
}
2.2.2 JS实现
//判断这个imei是否是正确的 正确返回ture 错误返回false
function isImei(imeiString) {
		let i = 0;
		let vl_Sum1 = 0, vl_Sum2 = 0, vl_Total = 0;
		let vl_Temp = 0;

		for (i = 0; i < 14; i++) {
			/*(1)将奇数位数字相加(从1开始计数)*/
			if ((i % 2) == 0) {
				vl_Sum1 = vl_Sum1 + parseInt(imeiString[i]);
			} else {
				/*(2)将偶数位数字分别乘以2,分别计算个位数和十位数之和(从1开始计数)*/
				vl_Temp = (parseInt(imeiString[i])) * 2;
				if (vl_Temp < 10) {
					vl_Sum2 = vl_Sum2 + vl_Temp;
				} else {
					vl_Sum2 = vl_Sum2 + 1 + vl_Temp - 10;
				}
			}
		}
		/*(1)+(2)*/
		vl_Total = vl_Sum1 + vl_Sum2;
		/*如果得出的数个位是0则校验位为0,否则为10减去个位数 */
		if ((vl_Total % 10) == 0) {
			if(imeiString[14] == '0'){
				return true;
			}
		} else {
			if((10 - (vl_Total % 10)).toString() == imeiString[14] ){
				return true;
			}
		}
		return false;
	}

以上两种算法都已验证正确,其他语言的算法大同小异,不再赘述。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值