上篇文章主要介绍《实现原理》,这篇看主要代码的编写。

实现IDGenerator类

  • 64位ID由以下元素组成:固定位占2位,时间戳占41位,服务实例数字编号占4位,业务编号占10位,自增id占7位
const BITS_FULL     = 64;
const BITS_PRE = 2; // 固定位01
const BITS_TIME = 41; // 可支持69年
const BITS_SERVER = 4; // 可支持16台集群服务
const BITS_WORKER = 10; // 可支持业务数1024个
const BITS_SEQUENCE = 7; // 一毫秒内支持生成128个id
  • 定义好三个变量:

    • $sequence_id: 自增序列id,在同一毫秒内产生的ID由此变量进行递增区分
    • $last_timestamp: 最后一次产生ID时的毫秒级时间戳,与下一次生成ID时的时间进行比较,如果时间相同则进行$sequence_id++,如果不同$sequence_id置0从新开始计算。
    • $server_id: 服务器id编号,代表不同的分布式服务实例,服务启动时指定。
private $sequence_id    = 0;
private $last_timestamp = 0;
private $server_id = 0;
  • EPOCH_TIME,由过去最近的一个时间值作为比较基数,如下值为2018-10-30 00:00:00
const EPOCH_TIME = 1540828800000; //时间基数
  • 获取一个64位ID主要由以下位运算完成
/**
* 获取id
*/
function get($worker_id = 0)
{
//初始化id位
$id = pow(2, 62); /* 1. 时间戳 41位 */
$time = $this->timeGen();
$diff_time = $time - self::EPOCH_TIME; $shift = self::BITS_FULL - self::BITS_PRE - self::BITS_TIME;
$id |= $diff_time << $shift; /* 2. 服务器id */
$shift -= self::BITS_SERVER;
$id |= ($this->server_id & (pow(2, self::BITS_SERVER) - 1)) << $shift; /* 3. 业务id */
$shift -= self::BITS_WORKER;
$id |= ($worker_id & (pow(2, self::BITS_WORKER) - 1)) << $shift; /* 4. 自增id */
$id |= ($this->sequence_id % (pow(2, self::BITS_SEQUENCE) - 1)); $this->sequence_id++;
return $id;
} /**
* 获取当前时间
*/
function timeGen() {
$wait_next_ms = 0;
do {
if($wait_next_ms > 0) {
usleep(100); //等待下一毫秒,休眠0.1毫秒
} $timestamp = microtime(true) * 1000;
$timestamp = (int) $timestamp; if($this->last_timestamp < $timestamp) {
$this->sequence_id = 0;
} $wait_next_ms++; } while ($this->last_timestamp == $timestamp
&& $this->sequence_id >= (pow(2, self::BITS_SEQUENCE) - 1)
|| $this->last_timestamp > $timestamp); $this->last_timestamp = $timestamp;
return $timestamp;
}

实现Swoole redis服务

$shortopts = 's:p'; // s服务ID编号, p绑定端口
$options = getopt($shortopts); // 由swoole_table存储最后一次产生数据的相关值
$table = new swoole_table(2048);
$table->column('sequence_id', swoole_table::TYPE_INT);
$table->column('last_timestamp', swoole_table::TYPE_FLOAT);
$table->column('server_id', swoole_table::TYPE_INT);
$table->create(); // atomic分业务加锁
for ($i = 0; $i <= 1023; $i++) {
$atomics[$i] = new swoole_atomic(0);
} $serv = new Server("0.0.0.0", $options['p']);
$serv->table = $table;
$serv->atomics = $atomics; $IDGen_config = array(
'server_id' => $options['s'],
'last_timestamp' => 0,
'sequence_id' => 0,
);
$serv->table->set('key:idgen_config', $IDGen_config); $serv->on('start', function ($serv) use($options) {
// 启动事件
}); //监听redis get指令
$serv->setHandler('GET', function ($fd, $data) use ($serv) {
$data = trim($data[0]);
$IDGen = new IDGen();
$id = $IDGen->get(); return Server::format(Server::STRING, $id);
}); $serv->start();

Redis-cli连接测试

redis-cli -h 172.19.19.21 -p 9500
172.19.19.21:9500> get 1
"4747443928557682816"
172.19.19.21:9500> get 2
"4747443933230137600"

分布式ID生成器PHP+Swoole实现(下) - 代码实现的更多相关文章

  1. 分布式ID生成器PHP+Swoole实现(上) - 实现原理

    1.发号器介绍 什么是发号器? 全局唯一ID生成器,主要用于分库分表唯一ID,分布式系统数据的唯一标识. 是否需要发号器? 1)是否需要全局唯一. 分布式系统应该不受单点递增ID限制,中心式的会涉及到 ...

  2. c#分布式ID生成器

    c#分布式ID生成器   简介 这个是根据twitter的snowflake来写的.这里有中文的介绍. 如上图所示,一个64位ID,除了最左边的符号位不用(固定为0,以保证生成的ID都是正数),还剩余 ...

  3. 来吧,自己动手撸一个分布式ID生成器组件

    在经过了众多轮的面试之后,小林终于进入到了一家互联网公司的基础架构组,小林目前在公司有使用到架构组研究到分布式id生成器,前一阵子大概看了下其内部的实现,发现还是存在一些架构设计不合理之处.但是又由于 ...

  4. 分布式ID生成器(CosId)的设计与实现

    分布式ID生成器(CosId)设计与实现 CosId 简介 CosId 旨在提供通用.灵活.高性能的分布式 ID 生成器. 目前提供了俩类 ID 生成器: SnowflakeId : 单机 TPS 性 ...

  5. 分布式ID生成器及redis,etcd分布式锁

    分布式id生成器 有时我们需要能够生成类似MySQL自增ID这样不断增大,同时又不会重复的id.以支持业务中的高并发场景.比较典型的,电商促销时,短时间内会有大量的订单涌入到系统,比如每秒10w+.明 ...

  6. 常用的分布式ID生成器

    为何需要分布式ID生成器 **本人博客网站 **IT小神 www.itxiaoshen.com **拿我们系统常用Mysql数据库来说,在之前的单体架构基本是单库结构,每个业务表的ID一般从1增,通过 ...

  7. go语言实现分布式id生成器

    本文:https://chai2010.cn/advanced-go-programming-book/ch6-cloud/ch6-01-dist-id.html 分布式id生成器 有时我们需要能够生 ...

  8. 分布式ID生成器的解决方案总结

    在互联网的业务系统中,涉及到各种各样的ID,如在支付系统中就会有支付ID.退款ID等.那一般生成ID都有哪些解决方案呢?特别是在复杂的分布式系统业务场景中,我们应该采用哪种适合自己的解决方案是十分重要 ...

  9. CosId 通用、灵活、高性能的分布式 ID 生成器

    CosId 通用.灵活.高性能的分布式 ID 生成器 介绍 CosId 旨在提供通用.灵活.高性能的分布式系统 ID 生成器. 目前提供了俩大类 ID 生成器:SnowflakeId (单机 TPS ...

随机推荐

  1. Word2007文档中怎么输入上标下标

    1.Word中输出Z = X2 + Y2 此公式流程: 首先在Word中写入:Z = X2 + Y2: 方法1:选中X后面的2,再按组合键“Ctrl+Shift+加号键”即可,如此操作Y后面的2即可.

  2. C++多线程编程一

    1.C++多线程初步: #include <iostream> #include <thread> #include <Windows.h> using names ...

  3. 2016级算法期末上机-B.简单·ModricWang's Fight with DDLs I

    1124 ModricWang's Fight with DDLs I 思路 这道题本质上就是一个多项式求值,题目中的n需要手动算一下,单位复根可以根据复数的性质来求,即\(e^{i\pi}+1=0\ ...

  4. jQuery css()与class()的用法

    一.css()用法:   1.设置css // css(name, value) // 修改单个样式 // name:样式名 value:样式值 $("li") .css(&quo ...

  5. mysql工具——mysqlcheck(MYISAM)

    基本介绍 演示: 使用optimize的时候,可能会出现 Table does not support optimize, doing recreate + analyze instead 这时候参考 ...

  6. SpringBoot 整合 中国移动 MAS HTTP1.0 实现短信发送服务

    因为客户需要,本身使用的 阿里云的短信服务改为了中国移动MAS HTTP 1.0  短信通知,因为看到网络上关于此类的博客知识很少,再趟完坑后特地写下这篇博客,提醒后来人. 特别感谢 中国移动MAS ...

  7. UNIX基础概念

    UNIX基本概念 进程 从用户观点来看:进程是程序的一个执行实例. 从UNIX系统内部来看,是为运行程序提供执行环境的实体,是系统进行资源分配和调度运行的一个单位. 进程有三个特点: 1)进程有一个控 ...

  8. 服务器反爬虫攻略:Apache/Nginx/PHP禁止某些User Agent抓取网站

    我们都知道网络上的爬虫非常多,有对网站收录有益的,比如百度蜘蛛(Baiduspider),也有不但不遵守robots 规则对服务器造成压力,还不能为网站带来流量的无用爬虫,比如宜搜蜘蛛(YisouSp ...

  9. Focal Loss 的前向与后向公式推导

    把Focal Loss的前向和后向进行数学化描述.本文的公式可能数学公式比较多.本文尽量采用分解的方式一步一步的推倒.达到能易懂的目的. Focal Loss 前向计算 其中 是输入的数据 是输入的标 ...

  10. 【App性能监控】:Android studio环境的搭建(以及遇到个各种坑)

    今天搭建app性能测试环境,使用的是android studio的Android Device Monitor抓取trace日志分析: 1,下载最新的android studio安装,这一步没啥问题: ...