代码如下所示:
我们用0-Z(0123456789ABCDEFGHIJKLMNOPQRSTUVWXYZ)来表示0-35的值,比如字母Z代表35。在这种情况下php进制转换,我想得到一个5位数字,最大信息量是36的5次方,36^5=60466176,即最大的5位数字相当于10补数:60466176。
为了在本文中进行演示,我们假设俱乐部收到一批 10 位会员卡号。 会员卡号由3位城市代码+5位卡号代码+2位校准代码组成。 城市号码用区号表示,如上海为755,5位卡号由36位补码组成。 前两个校准码是通过一定的算法生成的。 校准码的优点是可以验证卡号的合法性。 这样的话,我们生成的10位卡号相当于最多超过6000万个会员卡号,而且是唯一不重复的卡号。
我们使用PHP进行补码转换,10的补码到36的补码。
class Code {
//密码字典
private $dic = array(
0=>'0', 1=>'1', 2=>'2', 3=>'3', 4=>'4', 5=>'5', 6=>'6', 7=>'7', 8=>'8',
9=>'9', 10=>'A', 11=>'B', 12=>'C', 13=>'D', 14=>'E', 15=>'F', 16=>'G', 17=>'H',
18=>'I',19=>'J', 20=>'K', 21=>'L', 22=>'M', 23=>'N', 24=>'O', 25=>'P', 26=>'Q',
27=>'R',28=>'S', 29=>'T', 30=>'U', 31=>'V', 32=>'W', 33=>'X', 34=>'Y', 35=>'Z'
);
public function encodeID($int, $format=8) {
$dics = $this->dic;
$dnum = 36; //进制数
$arr = array ();
$loop = true;
while ($loop) {
$arr[] = $dics[bcmod($int, $dnum)];
$int = bcdiv($int, $dnum, 0);
if ($int == '0') {
$loop = false;
}
}
if (count($arr) < $format)
$arr = array_pad($arr, $format, $dics[0]);
return implode('', array_reverse($arr));
}
public function decodeID($ids) {
$dics = $this->dic;
$dnum = 36; //进制数
//键值交换
$dedic = array_flip($dics);
//去零
$id = ltrim($ids, $dics[0]);
//反转
$id = strrev($id);
$v = 0;
for ($i = 0, $j = strlen($id); $i < $j; $i++) {
$v = bcadd(bcmul($dedic[$id {
$i }
], bcpow($dnum, $i, 0), 0), $v, 0);
}
return $v;
}
}
我们定义Code类,首先定义密码字典,即分别对应0-Z的值,技能encodeID($int, $format)中参数$int代表数字,$format表示位数,例如encodeID(123456789, 5)表示将数字123456789转换为5位36的补码,而decodeID($ids)则用于将36的补码转换为10的补码。
我们可以这样生成卡号:
$code = new Code();
$card_no = $code->encodeID(888888,5);
如上,我们可以得到一个5位的卡号,它实际上代表的是一个卡号为888888(6个8)的会员号,而实际转换后就是一个5位的数字:0J1VC。
然后,我们添加城市号码和校准代码。 城市号码已经定义好,校准码是通过一定的算法得到的。 在这个例子中,我们使用一个简单的算法:将城市号的前三位和卡号的五位相加进行md5加密,然后将md5值的前2位作为校准码,从而得到数字前面的两位校准代码。
$card_pre = '755';
$card_vc = substr(md5($card_pre.$card_no),0,2);
$card_vc = strtoupper($card_vc);
echo $card_pre.$card_no.$card_vc;
在实际应用中,可以通过数据库获取10位补码php进制转换,以保证该号码的唯一性,然后将上述代码进行组合,最终生成10位唯一的会员卡号。
功效图:
本文来自: