PHP实现Snowflake生成分布式唯一ID的方法示例


Posted in PHP onAugust 30, 2020

前言

Twitter 的 snowflake 在分布式生成唯一 UUID 应用还是蛮广泛的,基于 snowflake 的一些变种的算法网上也有不少。使用 snowflake 生成 UUID 很多都是在分布式场景下使用,我看了下网上有其中有几篇 PHP 实现的都没有考虑到线程安全。现在 PHP 有了 Swoole 的锁和协程的加持,对于我们开发线程安全和高并发模拟还是很方便的,这里用 PHP 结合 Swoole 来学习下实现最简单的 snowflake。

先来看以下 snowflake 的结构:

PHP实现Snowflake生成分布式唯一ID的方法示例

生成的数值是 64 位,分成 4 个部分:

  • 第一个 bit 为符号位,最高位为 0 表示正数
  • 第二部分 41 个 bit 用于记录生成 ID 时候的时间戳,单位为毫秒,所以该部分表示的数值范围为 2^41 - 1(69 年),它是相对于某一时间的偏移量
  • 第三部分的 10 个 bit 表示工作节点的 ID,表示数值范围为 2^10 - 1,相当于支持 1024 个节点
  • 第四部分 12 个 bit 表示每个工作节点没毫秒生成的循环自增 id,最多可以生成 2^12 -1 个 id,超出归零等待下一毫秒重新自增。
<?php

class Snowflake
{
  const EPOCH = 1543223810238;  // 起始时间戳,毫秒

  const SEQUENCE_BITS = 12;  //序号部分12位
  const SEQUENCE_MAX = -1 ^ (-1 << self::SEQUENCE_BITS); // 序号最大值

  const WORKER_BITS = 10; // 节点部分10位
  const WORKER_MAX = -1 ^ (-1 << self::WORKER_BITS); // 节点最大数值

  const TIME_SHIFT = self::WORKER_BITS + self::SEQUENCE_BITS; // 时间戳部分左偏移量
  const WORKER_SHIFT = self::SEQUENCE_BITS;  // 节点部分左偏移量

  protected $timestamp;  // 上次ID生成时间戳
  protected $workerId;  // 节点ID
  protected $sequence;  // 序号
  protected $lock;    // Swoole 互斥锁

  public function __construct($workerId)
  {
    if ($workerId < 0 || $workerId > self::WORKER_MAX) {
      trigger_error("Worker ID 超出范围");
      exit(0);
    }

    $this->timestamp = 0;
    $this->workerId = $workerId;
    $this->sequence = 0;
    $this->lock = new swoole_lock(SWOOLE_MUTEX);
  }

  /**
   * 生成ID
   * @return int
   */
  public function getId()
  {
    $this->lock->lock();  // 这里一定要记得加锁
    $now = $this->now();
    if ($this->timestamp == $now) {
      $this->sequence++;

      if ($this->sequence > self::SEQUENCE_MAX) {
        // 当前毫秒内生成的序号已经超出最大范围,等待下一毫秒重新生成
        while ($now <= $this->timestamp) {
          $now = $this->now();
        }
      }
    } else {
      $this->sequence = 0;
    }

    $this->timestamp = $now;  // 更新ID生时间戳

    $id = (($now - self::EPOCH) << self::TIME_SHIFT) | ($this->workerId << self::WORKER_SHIFT) | $this->sequence;
    $this->lock->unlock(); //解锁

    return $id;
  }

  /**
   * 获取当前毫秒
   * @return string
   */
  public function now()
  {
    return sprintf("%.0f", microtime(true) * 1000);
  }

}

其实逻辑并不复杂,解释一下代码中的位运算:

-1 ^ (-1 << self::SEQUENCE_BITS)
就是-1的二进制表示为1的补码,其实等同于 :
2**self::SEQUENCE_BITS - 1

最后部分左移后或运算:

(($now - self::EPOCH) << self::TIME_SHIFT) | ($this->workerId << self::WORKER_SHIFT) | $this->sequence;

这里主要是对除了第一位符号位以外的三个部分进行左移相应的偏移量使其归位,并通过或运算重新整合成上面 snowflake 的结构,比如我们用 3 部分 4 位来演示一下该归并操作:

0000 0000 0010  --左移0位--> 0000 0000 0010
0000 0000 0100  --左移4位--> 0000 0100 0000 --或操作-->1000 0100 0010
0000 0000 1000  --左移8位--> 1000 0000 0000

总结

到此这篇关于PHP实现Snowflake生成分布式唯一ID的文章就介绍到这了,更多相关PHP Snowflake生成分布式唯一ID内容请搜索三水点靠木以前的文章或继续浏览下面的相关文章希望大家以后多多支持三水点靠木!

PHP 相关文章推荐
Discuz 5.0 中读取纯真IP数据库函数分析
Mar 16 PHP
一步一步学习PHP(4) php 函数 补充2
Feb 15 PHP
同台服务器使用缓存APC效率高于Memcached的演示代码
Feb 16 PHP
使用cookie实现统计访问者登陆次数
Jun 08 PHP
PHP获取和操作配置文件php.ini的几个函数介绍
Jun 24 PHP
利用php+mcDropdown实现文件路径可在下拉框选择
Aug 07 PHP
分享下页面关键字抓取www.icbase.com站点代码(带asp.net参数的)
Jan 30 PHP
php顺序查找和二分查找示例
Mar 27 PHP
php清空(删除)指定目录下的文件,不删除目录文件夹的实现代码
Sep 04 PHP
thinkphp实现like模糊查询实例
Oct 29 PHP
PHP防止刷新重复提交页面的示例代码
Nov 11 PHP
ThinkPHP+EasyUI之ComboTree中的会计科目树形菜单实现方法
Jun 09 PHP
Yii实现微信公众号场景二维码的方法实例
Aug 30 #PHP
Swoole源码中如何查询Websocket的连接问题详解
Aug 30 #PHP
PHP常用header头定义代码示例汇总
Aug 29 #PHP
PHP isset()及empty()用法区别详解
Aug 29 #PHP
PHP实现简单日历类编写
Aug 28 #PHP
PHP实现文件上传与下载
Aug 28 #PHP
PHP实现计算器小功能
Aug 28 #PHP
You might like
Discuz!5的PHP代码高亮显示插件(黑暗中的舞者更新)
2007/01/29 PHP
PHP实现的博客欢迎提示功能(很特别哦)
2014/06/05 PHP
thinkphp修改配置进入默认首页的方法
2017/02/07 PHP
OAuth认证协议中的HMACSHA1加密算法(实例)
2017/10/25 PHP
Javascript 面向对象之重载
2010/05/04 Javascript
setTimeout的延时为0时多个浏览器的区别
2012/05/23 Javascript
JS实现随机数生成算法示例代码
2013/08/08 Javascript
jquery 为a标签绑定click事件示例代码
2014/06/23 Javascript
jQuery实现伪分页的方法分享
2016/02/17 Javascript
confirm确认对话框的实现方法总结
2016/06/17 Javascript
详细谈谈javascript的对象
2016/07/31 Javascript
jquery.validate[.unobtrusive]和Bootstrap实现tooltip错误提示问题分析
2016/10/30 Javascript
Bootstrap导航条鼠标悬停下拉菜单
2017/01/04 Javascript
vue分类筛选filter方法简单实例
2017/03/30 Javascript
react-native ListView下拉刷新上拉加载实现代码
2017/08/03 Javascript
angularjs 缓存的使用详解
2018/03/19 Javascript
JavaScript折半查找(二分查找)算法原理与实现方法示例
2018/08/06 Javascript
Python中条件选择和循环语句使用方法介绍
2013/03/13 Python
Python下的Mysql模块MySQLdb安装详解
2014/04/09 Python
Python获取Windows或Linux主机名称通用函数分享
2014/11/22 Python
Python eval的常见错误封装及利用原理详解
2019/03/26 Python
selenium获取当前页面的url、源码、title的方法
2019/06/12 Python
在Python中画图(基于Jupyter notebook的魔法函数)
2019/10/28 Python
python爬虫请求头设置代码
2020/07/28 Python
Python中正则表达式对单个字符,多个字符和匹配边界等使用
2021/01/27 Python
常用UNIX 命令(Linux的常用命令)
2015/12/26 面试题
仓库理货员岗位职责
2013/12/18 职场文书
红领巾广播站广播稿
2014/02/01 职场文书
四个太阳教学反思
2014/02/01 职场文书
六年级学生评语大全
2014/12/26 职场文书
2015年艾滋病防治工作总结
2015/05/22 职场文书
观看禁毒宣传片后的感想
2015/08/11 职场文书
乡镇团代会开幕词
2016/03/04 职场文书
python爬虫--selenium模块
2021/03/31 Python
Java 数组内置函数toArray详解
2021/06/28 Java/Android
VMware虚拟机安装 Windows Server 2022的详细图文教程
2022/09/23 Servers