php如何利用32進位實現對id的加密解密(附代碼)

來源:互聯網
上載者:User
這篇文章給大家介紹的內容是關於php如何利用32進位實現對id的加密解密(附代碼),有一定的參考價值,有需要的朋友可以參考一下,希望對你有所協助。

最近在項目中遇到一個問題,目前使用者分享一個邀請碼給好友,好友根據邀請碼註冊成為新使用者之後,則成為目前使用者的下級,特定條件下,可以得到下級使用者的一系列返利。這裡要實現的就是根據目前使用者的id,產生一個加密串,並且可以反向解密。經過不斷的測試調整,終於得到了最後的結果。如:

id = 12 code = 85U43DM

初次實現

先上代碼,如下:

/** * 加密解密使用者邀請碼, * @param unknown $string * @param string $action  encode|decode * @return string */function endecodeUserId($string, $action = 'encode') {    $startLen = 13;    $endLen = 8;    $coderes = '';    #TOD 暫設定uid字元長度最大到9    if ($action=='encode') {        $uidlen = strlen($string);        $salt = 'yourself_code';        $codestr = $string.$salt;        $encodestr = hash('md4', $codestr);        $coderes = $uidlen.substr($encodestr, 5,$startLen-$uidlen).$string.substr($encodestr, -12,$endLen);        $coderes = strtoupper($coderes);    }elseif($action=='decode'){        $strlen = strlen($string);        $uidlen = $string[0];        $coderes = substr($string, $startLen-$uidlen+1,$uidlen);    }    return  $coderes;}

思路介紹:

設定一個鹽值,$salt,和id拼接後組成一個新的字串,該鹽值可用於後期對邀請碼進行安全校正。對該字串進行md4加密(考慮到相比md5,md4速度更快,並且安全性也並不弱),得到$encodestr,對該字串進行拆分,分為前後兩部分,第一部分$startLen,13個字串;第二部分$endLen,8個字串。將$string,這裡指傳入的id,和$uidlen,混入前一部分字串。因這裡目前僅支援id最大長度為9,因此$uidlen長度為1,這樣最後我們便得到了一個長度為22的字串。

加密的過程中,我們實際上是把id的數值和id的長度,混入到了加密串中,加密的時候我們根據存入的這些資訊找到對應的位置,即可得到id。

這裡,我們對安全性並沒有要求很高,為了使程式運行速度更快,因此在解密的時候並沒有驗證。

測試,對id加密:

echo endecodeUserId(12);

輸出結果:

23471DC2352712F34D6780

測試,對邀請碼解密

 echo endecodeUserId('23471DC2352712F34D6780','decode');

輸出結果:

12

得到的結果看上去並沒有問題,但是實際測試中發現這樣一個問題,對於普通使用者可能會存在這種情況,好友發到他手機上一個邀請碼,然後他想要用電腦進行註冊,但他並不知道該怎麼樣把邀請碼從手機傳到電腦上或者嫌麻煩,這時候他就要在電腦開始手動輸入邀請碼了,天哪,22位啊,還是大寫字母加數字混合,估計他要放棄註冊了。

因此,我們進行了調整,改成7位的邀請碼。

再次探索

這裡是在寫文章之前對方法進行了封裝,還是直接先上代碼

<?phpclass convert{    /**     * 初始數字,自訂     */    const INIT_NUM = 123456789;    /**     * @var 進位的基底字元串     */    private $baseChar;    /**     * @var 進位類型     */    private $type;    /**     * @var array 各進位字串列表     */    private static $convertList = array(        '32' => '0123456789ABCDEFGHJKMNPQRSTVWXYZ',//不含ILOU    );    public function __construct($type='32')    {        $this->type = $type;        $this->baseChar = self::$convertList[$type];    }    /**     * 公用方法,數字進行進位轉換     * @param $num     * @return string     */    private function _idToString($num){        $str = '';        while ($num!=0){            $tmp = $num % $this->type;            $str .= $this->baseChar[$tmp];            $num = intval($num/$this->type);        }        return $str;    }    /**     * @desc  im:十機制數轉換成三十位元     * @param (string)$char 三十位元     * return 返回:十進位數     */    public function idToString($id){//10位內id 返回7位字母數字        //數組 增加備用數值        $id += self::INIT_NUM;        //左補0 補齊10位        $str = str_pad($id,10,'0',STR_PAD_LEFT);        //按位 拆分 4 6位(32進位 4 6位劃分)        $num1 = intval($str[0].$str[2].$str[6].$str[9]);        $num2 = intval($str[1].$str[3].$str[4].$str[5].$str[7].$str[8]);        $str1 = $str2 = '';        $str1 = $this->_idToString($num1);        $str1 = strrev($str1);        $str2 = $this->_idToString($num2);        $str2 = strrev($str2);        //4 補足 3 4位 U L        return str_pad($str1,3,'U',STR_PAD_RIGHT).str_pad($str2,4,'L',STR_PAD_RIGHT);    }    /**     * @desc  im:三十位元轉換成十機制數     * @param (string)$char 三十位元     * return 返回:十進位數     */    public function stringToId($str){        //1 清除 3 4 位補足位        $str1 = trim(substr($str,0,3),'U');        $str2 = trim(substr($str,3,4),'L');        $num1 = $this->_stringToId($str1);        $num2 = $this->_stringToId($str2);        //補位拼接        $str1 = str_pad($num1,4,'0',STR_PAD_LEFT);        $str2 = str_pad($num2,6,'0',STR_PAD_LEFT);        $id = ltrim($str1[0].$str2[0].$str1[1].$str2[1].$str2[2].$str2[3].$str1[2].$str2[4].$str2[5].$str1[3],'0');        //減去 備用數值        $id -= self::INIT_NUM;        return $id;    }    /**     * 公用方法字串轉數字     * @param $str     * @return float|int|string     */    private function _stringToId($str){        //轉換為數組        $charArr = array_flip(str_split($this->baseChar));        $num = 0;        for ($i=0;$i<=strlen($str)-1;$i++)        {            $linshi = substr($str,$i,1);            if(!isset($charArr[$linshi])){                return '';            }            $num += $charArr[$linshi]*pow($this->type,strlen($str)-$i-1);        }        return $num;    }}

思路介紹

在一位工作多年的大神的指導下,採用了這種方法。將id轉化為固定長度的32進位字串,並加上自己的演算法。為什麼這裡採用32進位,而不是其他進位呢?32進位可以包含足夠多的英文字元,產生的加密串看起來會更規範,另一方面,排除一些不容易識別的英文字元(這裡排除ILOU),因此採用了32進位,而並沒有採用36進位。

加密過程,方法idToString(),因考慮到剛開始id比較小的時候,轉為32進位會出現比較多的0,看起來很不規範,因此設定一個初始值INIT_NUM,這個可以自訂。根據傳過來的id,加上初始值後得到一個長度為10位的數值,將這個數值間隔位拆開分為長度為4位的$num1和長度為6位的$num2,兩個數值分別轉換為32進位,$num1轉化後得到長度為3的字串,不足的用U補足,$num2得到長度為4的字串,不足的用L來補足。

解密則是逆操作,反向操作即可。

測試:產生

$obj = new convert(32);$res1 = $obj->idToString(12);

結果:

85U43DM

解密:

$obj = new convert(32);$res1 = $obj->stringToId('85U43DM');

結果:

12

相關文章

聯繫我們

該頁面正文內容均來源於網絡整理,並不代表阿里雲官方的觀點,該頁面所提到的產品和服務也與阿里云無關,如果該頁面內容對您造成了困擾,歡迎寫郵件給我們,收到郵件我們將在5個工作日內處理。

如果您發現本社區中有涉嫌抄襲的內容,歡迎發送郵件至: info-contact@alibabacloud.com 進行舉報並提供相關證據,工作人員會在 5 個工作天內聯絡您,一經查實,本站將立刻刪除涉嫌侵權內容。

A Free Trial That Lets You Build Big!

Start building with 50+ products and up to 12 months usage for Elastic Compute Service

  • Sales Support

    1 on 1 presale consultation

  • After-Sales Support

    24/7 Technical Support 6 Free Tickets per Quarter Faster Response

  • Alibaba Cloud offers highly flexible support services tailored to meet your exact needs.