需求是每个用户都有个邀请码,邀请字母数字组成的10位字符串,首先想到的就是生成随机字符串了,如下
public function createNonceStr($length = 16)
{
$chars = 'abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ0123456789';
$str = '';
for ($i = 0; $i < $length; $i++) {
$str .= substr($chars, mt_rand(0, strlen($chars) - 1), 1);
}
return $str;
}
但是这种方法理论上还是存在相同的可能性,尤其是用户数量足够大的情况。
这里我找到了一种比较简单但是能保证唯一性的算法,而且生成的字符串长度也很小(用户id为99999999时生成加密字符串为gVKJn),8位数的用户ID仅生成了5位数的加密字符串,如下
/**
* 生成邀请码
* @param $userId 用户ID
*/
private function getInviteCode($userId)
{
$chars = 'abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ0123456789';
$num = strlen($chars);
$str = '';
while ($userId > 0) {
$mod = $userId % $num;
$userId = ($userId - $mod) / $num;
$str = $chars[$mod] . $str;
}
return $str;
}
为了达到格式的统一性,也为了展示出不确定性,可以在将第二个方法和第一个方法结合。当第二张方法生成的加密字符串位数不足时,用第一个方法生成的随机字符串拼接在其前方,如下要求邀请码为10位时的情况
/**
* 生成邀请码
* @param $userId 用户ID
*/
private function getInviteCode($userId)
{
$chars = 'abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ0123456789';
$num = strlen($chars);
$str = '';
while ($userId > 0) {
$mod = $userId % $num;
$userId = ($userId - $mod) / $num;
$str = $chars[$mod] . $str;
}
// 不足用随机字符串补充,10表示邀请码邀请10位
$str = str_pad($str, 10, self::createNonceStr(10 - strlen($str)), STR_PAD_LEFT);
return $str;
}