雪花算法
<?php
namespace App\Services;
/**
* 分布式 id 生成类 组成: <毫秒级时间戳+机器id+序列号>
* 默认情况下41bit的时间戳可以支持该算法使用到2082年,10bit的工作机器id可以支持1023台机器,
*序列号支持1毫秒产生4095个自增序列id
* @author zhangqi
*/
class IdCreate
{
const EPOCH = 1479533469598; //开始时间,固定一个小于当前时间的毫秒数\
const max12bit = 4095;
const max41bit = 1099511627775;
static $machineId = null; // 机器id
public static function machineId($mId = 0)
{
self::$machineId = $mId;
}
public static function createOnlyId()
{
// 时间戳 42字节
$time = floor(microtime(true) * 1000);
// 当前时间 与 开始时间 差值
$time -= self::EPOCH;
// 二进制的 毫秒级时间戳
$base = decbin(self::max41bit + $time);
// 机器id 10 字节
if (!self::$machineId) {
$machineid = self::$machineId;
} else {
$machineid = str_pad(decbin(self::$machineId), 10, "0", STR_PAD_LEFT);\
}
// 序列数 12字节
$random = str_pad(decbin(mt_rand(0, self::max12bit)), 12, "0", STR_PAD_LEFT);\
// 拼接
$base = $base . $machineid . $random;
// 转化为 十进制 返回
return bindec($base);
}
}
调用 IdCreate::createOnlyId(1);
返回的值 // 2219944901359
返回的值 // 2220004015923
返回的值 // 2220062766757
返回的值 // 2220119764908
本作品采用《CC 协议》,转载必须注明作者和本文链接
“机器id 10 字节”这个里面的提到的字节应称为位。还有最后一串的随机数组成,无法保证绝对的唯一性