基于php雪花算法工具类Snowflake -来自chatGPT

php 复制代码
<?php

class Snowflake {
    // 定义Snowflake算法的各个参数
    private $workerIdBits = 5;
    private $datacenterIdBits = 5;
    private $sequenceBits = 12;

    private $workerIdShift;
    private $datacenterIdShift;
    private $timestampLeftShift;

    private $maxWorkerId;
    private $maxDatacenterId;
    private $sequenceMask;

    private $workerId;
    private $datacenterId;
    private $sequence = 0;
    private $lastTimestamp = -1;

    public function __construct($workerId, $datacenterId) {
        // 计算位偏移量
        $this->workerIdShift = $this->sequenceBits;
        $this->datacenterIdShift = $this->sequenceBits + $this->workerIdBits;
        $this->timestampLeftShift = $this->sequenceBits + $this->workerIdBits + $this->datacenterIdBits;

        // 计算最大ID
        $this->maxWorkerId = -1 ^ (-1 << $this->workerIdBits);
        $this->maxDatacenterId = -1 ^ (-1 << $this->datacenterIdBits);
        $this->sequenceMask = -1 ^ (-1 << $this->sequenceBits);

        // 初始化参数
        $this->workerId = $workerId;
        $this->datacenterId = $datacenterId;
    }

    // 生成下一个唯一ID
    public function generateId() {
        // 获取当前时间戳(毫秒级)
        $timestamp = floor(microtime(true) * 1000);

        // 如果当前时间小于上次生成ID的时间戳,则抛出异常
        if ($timestamp < $this->lastTimestamp) {
            throw new Exception("Invalid system clock!");
        }

        // 如果当前时间戳与上次时间戳相同,则自增序列号
        if ($timestamp == $this->lastTimestamp) {
            $this->sequence = ($this->sequence + 1) & $this->sequenceMask;

            // 如果序列号等于0,则需要进入下一毫秒重新生成ID
            if ($this->sequence == 0) {
                $timestamp = $this->waitNextMillis($this->lastTimestamp);
            }
        } else {
            $this->sequence = 0;
        }

        // 保存最后生成ID的时间戳
        $this->lastTimestamp = $timestamp;

        // 生成最终的唯一ID
        $uniqueId = (($timestamp << $this->timestampLeftShift) |
                     ($this->datacenterId << $this->datacenterIdShift) |
                     ($this->workerId << $this->workerIdShift) |
                     $this->sequence);

        return $uniqueId;
    }

    // 阻塞到下一个毫秒,直到获得新的时间戳
    private function waitNextMillis($lastTimestamp) {
        $timestamp = floor(microtime(true) * 1000);
        while ($timestamp <= $lastTimestamp) {
            usleep(1000);
            $timestamp = floor(microtime(true) * 1000);
        }
        return $timestamp;
    }
}

// 测试方法
$snowflake = new Snowflake(1, 1);
for ($i = 0; $i < 10; $i++) {
    $uniqueId = $snowflake->generateId();
    echo $uniqueId . "\n";
}

?>

在上述代码中,我们创建了一个名为Snowflake的类,该类用于生成Snowflake算法的唯一ID。它具有以下主要属性和方法:

  • __construct($workerId, $datacenterId):初始化Snowflake算法的参数,包括工作节点ID(workerId)和数据中心ID(datacenterId)。
  • generateId():生成下一个唯一ID。
  • waitNextMillis($lastTimestamp):阻塞到下一个毫秒,直到获得新的时间戳。

在测试方法中,我们创建了一个Snowflake实例,并使用其generateId()方法生成了10个唯一ID,并打印出来。

相关推荐
fish_xk1 小时前
c++中的引用和数组
开发语言·c++
酒尘&4 小时前
JS数组不止Array!索引集合类全面解析
开发语言·前端·javascript·学习·js
冬夜戏雪4 小时前
【java学习日记】【2025.12.7】【7/60】
java·开发语言·学习
xwill*4 小时前
分词器(Tokenizer)-sentencepiece(把训练语料中的字符自动组合成一个最优的子词(subword)集合。)
开发语言·pytorch·python
咖啡の猫5 小时前
Python列表的查询操作
开发语言·python
Savior`L5 小时前
二分算法及常见用法
数据结构·c++·算法
quikai19816 小时前
python练习第三组
开发语言·python
JIngJaneIL6 小时前
基于Java非遗传承文化管理系统(源码+数据库+文档)
java·开发语言·数据库·vue.js·spring boot
吃西瓜的年年6 小时前
1. 初识C语言
c语言·开发语言
mmz12076 小时前
前缀和问题(c++)
c++·算法·图论