Laravel根据用户id生成四位数唯一邀请码
新建文件app/Services/InviteCodeService.php
<?php
namespace App\Services;
// 邀请码服务
class InviteCodeService
{
protected $key,$num;
public function __construct()
{
$this->key = 'abcdefghjkmnpqrstuvwxyz123456789';
// 注意这个key里面不能出现数字0 否则当 求模=0 会重复的
// 多少进制
$this->num = strlen($this->key);
}
// 传递用户id生成唯一邀请码
public function enCode(int $user_id)
{
$code = ''; // 邀请码
while ($user_id > 0) { // 转进制
$mod = $user_id % $this->num; // 求模
$user_id = ($user_id - $mod) / $this->num;
$code = $this->key[$mod] . $code;
}
$code = str_pad($code, 4, '0', STR_PAD_LEFT); // 不足用0补充
return $code;
}
// 邀请码获取用户id 一般都不需要用到
function deCode($code)
{
if (strrpos($code, '0') !== false)
$code = substr($code, strrpos($code, '0') + 1);
$len = strlen($code);
$code = strrev($code);
$user_id = 0;
for ($i = 0; $i < $len; $i++)
$user_id += strpos($this->key, $code[$i]) * pow($this->num, $i);
return $user_id;
}
}
绑定到容器
看爱好,如果不想绑定就 new \App\Services\InviteCodeService
。
编辑app/Providers/AppServiceProvider.php
use App\Services\InviteCodeService;
public function register()
{
$this->app->singleton('invite_code',InviteCodeService::class);
}
测试唯一性
$max_num = 200000;
$codes = [];
for ($i = 1; $i <= $max_num; $i++)
$codes[] = app('invite_code')->enCode($i);
$i = 1;
foreach ($codes as $code){
$userId = app('invite_code')->deCode($code); // 邀请码获取用户id
if( $userId != $i)
dd("邀请码解密错误".$i);
$i++;
}
$unique_count = count(array_unique($codes));
dd($unique_count); // 不重复的总数
20w
数据是没有重复的,并且邀请码解 用户id
也是正确的。
要注意的
$this->key
不能出现数字0
。$this->key
不能有重复的字符串。 如:abccd
c
重复。
关于$this->key
$this->key
顺序可以打乱的,// $this->key = 'abcdefghjkmnpqrstuvwxyz123456789'; // 没打乱的 $this->key = 'kh8sjpdazetnmb5yw7rq4gc9fuv3216x'; // 打乱的
$this->key
长度不限制, 但是最好别太短。
如abcd
: 最大的用户id
是4的3次方 (256)
超过256会怎么样? 邀请码变成5位而已 ……. ,不好看。为用户体验,
$this->key
字符串别加i
o
l
这些字母,因为容易混淆用户。i
:l
l
:1
o
:0
(当然0
是不能出现的)
很相似。
Ps
生成的邀请码位数 取决于 用户id
可以被 取模多少次。
例
假设$this->key = abcd
那就是4进制了。
如:用户id
255
邀请码4位数。
256
邀请码5位数。
500
邀请码5位数。
……
(所以对邀请码位数有限定的话,聪明的你会懂的怎么办的吧)
32
进制 邀请码是4位 最大用户id是多少?
32^4 = 1048576
扯淡
因为要做邀请码功能,第一反应:
百度搜出来代码虽然能用,有重复,不理想,就改编别人代码了。
然后就有这篇文章了。
本作品采用《CC 协议》,转载必须注明作者和本文链接
本帖由系统于 4年前 自动加精
高认可度评论:
给你推荐一个包
赞 :+1:
get 到了,但是不能白嫖,赞!
:+1: :+1: :+1:
赞 :+1:
蛮勤快啊~ :+1:
:+1:
:+1: :+1: :+1: :+1:
收藏了 :+1:
ma
32 位的话,最多是不是只能生成 32^4 个唯一字符?如果用户数超过这个值,会不会有重复,很好奇
给你推荐一个包
这个算法不错
点赞 :+1:
不知道对于Uuid 是否使用,出差中,等回去了我再试试
真厉害
Mark
base_convert 不香吗
简单修改了一下放到了 gist 上: gist.github.com/guanguans/52b9136e...
注意这个key里面不能出现数字0 否则当 求模=0 会重复的
为什么不能出现数字0,我理解这个key是唯一的就可以了,出现0和求模运算没有关系吧?
这儿可以这么优化一下
floor($user_d / $this->num)
看了一下,下面不足用0补充,如果有0的话,这儿这个0会影响解码结果。
nice~
真不错,点赞 :smile: