当前位置: 首页 > article >正文

雪花算法应用

什么是雪花算法?
雪花算法是由 Twitter 开源的分布式 ID 生成算法,用于生成 64 位的长整型唯一 ID。其结构如下:

- 1 位符号位:始终为 0
- 41 位时间戳:精确到毫秒
- 10 位工作机器 ID:包含 5 位数据中心 ID 和 5 位机器 ID
- 12 位序列号:同一毫秒内的自增序号
Golang 实现
以下是一个完整的 Golang 实现:

package snowflakeimport ("sync""time""errors"
)// Snowflake 结构体
type Snowflake struct {mutex         sync.Mutex    // 互斥锁lastTimestamp int64        // 上次的时间戳workerId      int64        // 工作节点IDdatacenterId  int64        // 数据中心IDsequence      int64        // 序列号
}const (workerBits     = uint(5)                              // 工作节点ID位数datacenterBits = uint(5)                              // 数据中心ID位数sequenceBits   = uint(12)                             // 序列号位数maxWorkerId     = -1 ^ (-1 << workerBits)            // 最大工作节点IDmaxDatacenterId = -1 ^ (-1 << datacenterBits)        // 最大数据中心IDmaxSequence     = -1 ^ (-1 << sequenceBits)          // 最大序列号timeShift      = workerBits + datacenterBits + sequenceBits    // 时间戳左移位数datacenterShift = workerBits + sequenceBits                    // 数据中心ID左移位数workerShift    = sequenceBits                                  // 工作节点ID左移位数epoch = int64(1672531200000) // 起始时间戳 (2023-01-01 00:00:00 +0800)
)// 创建新的雪花算法实例
func NewSnowflake(datacenterId, workerId int64) (*Snowflake, error) {if datacenterId > maxDatacenterId || datacenterId < 0 {return nil, errors.New("datacenter ID超出范围")}if workerId > maxWorkerId || workerId < 0 {return nil, errors.New("worker ID超出范围")}return &Snowflake{lastTimestamp: -1,workerId:      workerId,datacenterId:  datacenterId,sequence:      0,}, nil
}// 生成下一个ID
func (s *Snowflake) NextId() (int64, error) {s.mutex.Lock()defer s.mutex.Unlock()timestamp := time.Now().UnixMilli()// 时钟回拨检查if timestamp < s.lastTimestamp {return 0, errors.New("时钟回拨,拒绝生成ID")}// 同一毫秒内if timestamp == s.lastTimestamp {s.sequence = (s.sequence + 1) & maxSequence// 同一毫秒内序列号用完if s.sequence == 0 {// 等待下一毫秒for timestamp <= s.lastTimestamp {timestamp = time.Now().UnixMilli()}}} else {// 不同毫秒,序列号重置s.sequence = 0}s.lastTimestamp = timestamp// 组合IDid := ((timestamp - epoch) << timeShift) |(s.datacenterId << datacenterShift) |(s.workerId << workerShift) |s.sequencereturn id, nil
}
package mainimport ("fmt""log""your/path/snowflake"
)func main() {// 创建雪花算法实例(数据中心ID=1, 工作节点ID=1)sf, err := snowflake.NewSnowflake(1, 1)if err != nil {log.Fatal(err)}// 生成IDfor i := 0; i < 5; i++ {id, err := sf.NextId()if err != nil {log.Fatal(err)}fmt.Printf("ID %d: %d\n", i+1, id)}
}

 主要特点
1. 线程安全 :使用互斥锁确保并发安全
2. 高性能 :使用位运算进行计算
3. 唯一性 :通过时间戳+机器ID+序列号保证唯一
4. 有序性 :生成的ID整体趋势递增
5. 可配置 :支持自定义数据中心ID和工作节点ID
使用场景
1. 分布式系统中的全局唯一ID生成
2. 数据库分表分库的主键生成
3. 消息队列的消息ID
4. 订单号生成系统
5. 分布式日志追踪ID
注意事项
1. 确保服务器时间的准确性
2. 合理分配数据中心ID和工作节点ID
3. 注意时钟回拨问题的处理
4. 根据业务需求调整各部分位数分配
这个实现是线程安全的,并且适合在生产环境中使用。如果需要处理时钟回拨问题,可以根据具体需求添加相应的处理机制。

理时钟回拨问题处理

1. 等待方案
最基础的处理方式,当检测到时钟回拨时等待一段时间。

type Snowflake struct {mutex         sync.MutexlastTimestamp int64workerId      int64datacenterId  int64sequence      int64maxWaitTime   int64    // 最大等待时间(毫秒)
}func (s *Snowflake) NextId() (int64, error) {s.mutex.Lock()defer s.mutex.Unlock()timestamp := time.Now().UnixMilli()// 处理时钟回拨if timestamp < s.lastTimestamp {offset := s.lastTimestamp - timestampif offset <= s.maxWaitTime {// 等待时钟追上time.Sleep(time.Duration(offset) * time.Millisecond)timestamp = time.Now().UnixMilli()} else {return 0, errors.New("时钟回拨超过最大等待时间")}}// ... 后续逻辑保持不变
}

2. 备份时钟方案
使用备份时钟来处理回拨问题。

type BackupSnowflake struct {mutex         sync.MutexlastTimestamp int64workerId      int64datacenterId  int64sequence      int64backupDelta   int64    // 备份时钟递增步长
}func (s *BackupSnowflake) currentTimeMillis() int64 {timestamp := time.Now().UnixMilli()if timestamp < s.lastTimestamp {// 发生时钟回拨,使用备份时钟timestamp = s.lastTimestamp + s.backupDelta}return timestamp
}func (s *BackupSnowflake) NextId() (int64, error) {s.mutex.Lock()defer s.mutex.Unlock()timestamp := s.currentTimeMillis()if timestamp == s.lastTimestamp {s.sequence = (s.sequence + 1) & maxSequenceif s.sequence == 0 {timestamp = s.waitNextMillis(timestamp)}} else {s.sequence = 0}s.lastTimestamp = timestamp// ... 生成ID的逻辑
}

3. 号段预分配方案
预先分配一段序列号,降低对时间戳的依赖。

type SegmentSnowflake struct {mutex         sync.MutexlastTimestamp int64workerId      int64datacenterId  int64sequence      int64currentSegment int64segmentBits   uintsegmentSize   int64
}func NewSegmentSnowflake(datacenterId, workerId int64) *SegmentSnowflake {segmentBits := uint(12)return &SegmentSnowflake{datacenterId:   datacenterId,workerId:       workerId,segmentBits:    segmentBits,segmentSize:    1 << segmentBits,currentSegment: 0,}
}func (s *SegmentSnowflake) NextId() (int64, error) {s.mutex.Lock()defer s.mutex.Unlock()timestamp := time.Now().UnixMilli()if timestamp < s.lastTimestamp {// 发生时钟回拨,使用当前号段内的序列号if s.sequence < s.segmentSize-1 {s.sequence++return s.generateId(s.lastTimestamp, s.sequence), nil}// 当前号段用尽,切换到新的号段s.currentSegment++s.sequence = 0return s.generateId(s.lastTimestamp, s.currentSegment, s.sequence), nil}if timestamp != s.lastTimestamp {s.currentSegment = 0s.sequence = 0}s.lastTimestamp = timestampreturn s.generateId(timestamp, s.currentSegment, s.sequence), nil
}func (s *SegmentSnowflake) generateId(timestamp, segment, seq int64) int64 {return ((timestamp - epoch) << timeShift) |(s.datacenterId << datacenterShift) |(s.workerId << workerShift) |(segment << s.segmentBits) |seq
}

使用建议:

1. 对于短时间回拨(几毫秒到几秒),使用等待方案
2. 对于中等时间回拨,使用备份时钟方案
3. 对于长时间回拨或高并发场景,使用号段预分配方案
4. 在实际生产环境中,建议结合多种方案,并加入监控和告警机制
选择哪种方案需要根据具体的业务场景来决定:

- 如果业务对延迟敏感,避免使用等待方案
- 如果并发量大,考虑使用号段预分配方案
- 如果对时间戳精度要求高,可以使用备份时钟方案

相关文章:

雪花算法应用

什么是雪花算法&#xff1f; 雪花算法是由 Twitter 开源的分布式 ID 生成算法&#xff0c;用于生成 64 位的长整型唯一 ID。其结构如下&#xff1a; - 1 位符号位&#xff1a;始终为 0 - 41 位时间戳&#xff1a;精确到毫秒 - 10 位工作机器 ID&#xff1a;包含 5 位数据中心 …...

Chapter3:结构化程序设计

参考书籍&#xff1a;《C#边做边学》&#xff1b; 3.结构化程序设计 3.1 结构化程序设计的3种基本结构 顺序结构&#xff1a;先执行 A {\rm A} A语句&#xff0c;再执行 B {\rm B} B语句&#xff0c;两者是顺序执行的关系&#xff1b; 选择结构&#xff1a;根据所定选择条件为…...

白话文实战Nacos(保姆级教程)

前言 上一篇博客 我们创建好了微服务项目,本篇博客来体验一下Nacos作为注册中心和配置中心的功能。 注册中心 如果我们启动了一个Nacos注册中心,那么微服务比如订单服务,启动后就可以连上注册中心把自己注册上去,这过程就是服务注册。每个微服务,比如商品服务都应该注册…...

c语言函数学习

C语言函数学习笔记&#xff1a;从入门到实践 一、什么是函数&#xff1f; 函数是C语言中用于封装特定功能的代码块&#xff0c;是模块化编程的核心。通过函数可以实现&#xff1a; 代码复用&#xff1a;避免重复编写相同逻辑 逻辑清晰&#xff1a;将复杂程序分解为多个小模块…...

linux利用nfs服务器,实现数据和windows环境拷贝

1. 在Linux上设置NFS服务器 1.1 安装NFS服务器软件 首先&#xff0c;你需要在Linux服务器上安装NFS服务器软件。假设你使用的是基于Debian的系统&#xff08;如Ubuntu&#xff09;&#xff0c;可以按照以下步骤操作&#xff1a; sudo apt update sudo apt install nfs-kerne…...

智能理解 PPT 内容,快速生成讲解视频

当我们想根据一版 PPT 制作出相对应的解锁视频时&#xff0c;从撰写解锁词&#xff0c;录制音频到剪辑视频&#xff0c;每一个环节都需要投入大量的时间和精力&#xff0c;本方案将依托于阿里云函数计算 FC 和百炼模型服务&#xff0c;实现从 PPT 到视频的全自动转换&#xff0…...

FFmpeg + OpenGL ES 美颜相机教程大纲

做OpenGL和FFmpeg也有很长一段时间了&#xff0c;最近打算结合FFmpegOpenGL ES做一期视频教程&#xff0c;下面是完整视频教程大纲。最终的项目实战效果是实现一款美颜相机。教程分为理论讲解和实战开发两部分&#xff0c;适合有一定编程基础的开发者。课程计划是免费发布在B站…...

IEC61850标准下的数据和数据模型服务的详细介绍

目录 一、摘要 二、概述 三、详细介绍 1、读服务器目录(GetServerDirectory) 2、读逻辑设备目录(GetLogicalDeviceDirectory) 3、读逻辑节点目录(GetLogicalNodeDirectory) 4、读全部数据值(GetAllDataValues) 5、读数据值(GetDataValues) 6、设置数据值(SetDataValues…...

【3.Git与Github的历史和区别】

目录 Git的历史和Github的区别本质和功能 Git的历史和Github的区别 Git是由Linux内核的创造者Linus Torvalds于2005年创建的。当时&#xff0c;Linux内核开源项目使用BitKeeper作为版本控制系统&#xff0c;但2005年BitKeeper的商业公司终止了与Linux社区的合作&#xff0c;收…...

前端页面添加水印

前端页面添加水印 主要功能说明&#xff1a; 这是一个用于添加页面水印的工具函数水印会以半透明的形式显示在页面上&#xff0c;并且会重复平铺水印文字会有-15度的倾斜角度水印会覆盖整个页面&#xff0c;但不会影响页面的正常交互每次调用函数时会先删除已存在的水印&…...

Left side cannot be assigned to

Delphi XE E2064 Left side cannot be assigned to 错误解决方法-CSDN博客 Delphi XE E2064 Left side cannot be assigned to 错误解决方法 1. 起源 此问题源于[秋风人事档案管理系统]用Delphi XE重编译中所发现。 快十年了&#xff0c;当初Delphi 7所编写项目&#xff0c…...

R语言LCMM多维度潜在类别模型流行病学研究:LCA、MM方法分析纵向数据

全文代码数据&#xff1a;https://tecdat.cn/?p39710 在数据分析领域&#xff0c;当我们面对一组数据时&#xff0c;通常会有已知的分组情况&#xff0c;比如不同的治疗组、性别组或种族组等&#xff08;点击文末“阅读原文”获取完整代码数据&#xff09;。 然而&#xff0c;…...

伺服使能的含义解析

前言&#xff1a; 大家好&#xff0c;我是上位机马工&#xff0c;硕士毕业4年年入40万&#xff0c;目前在一家自动化公司担任软件经理&#xff0c;从事C#上位机软件开发8年以上&#xff01;我们在开发C#的运动控制程序的时候&#xff0c;一个必要的步骤就是对伺服上使能&#…...

ModuleJS 与 CommonJS 混用的两种解决方案

目录 方案一 方案二 统一使用 ModuleJS 统一使用CommonJS 方案一 使用构建工具&#xff0c;webpack、vite等系列构建工具。这些构建工具底层则会将两种不同的系统模块语言转为同一种语言&#xff0c;然后代码也能正常执行。 方案二 如果你可以修改文件的文件后缀&#xf…...

5. 【.NET 8 实战--孢子记账--从单体到微服务--转向微服务】--微服务基础工具与技术--Nacos

一、什么是Nacos Nacos 是阿里巴巴开源的一款云原生应用基础设施&#xff0c;它旨在简化微服务架构中服务治理和配置管理的复杂性。通过 Nacos&#xff0c;服务在启动时可以自动注册&#xff0c;而其他服务则可以通过名称来查找并访问这些注册好的实例。同时&#xff0c;Nacos…...

VUE项目中实现权限控制,菜单权限,按钮权限,接口权限,路由权限,操作权限,数据权限实现

VUE项目中实现权限控制&#xff0c;菜单权限&#xff0c;按钮权限&#xff0c;接口权限&#xff0c;路由权限&#xff0c;操作权限&#xff0c;数据权限实现 权限系统分类&#xff08;RBAC&#xff09;引言菜单权限按钮权限接口权限路由权限 菜单权限方案方案一&#xff1a;菜单…...

网站的记住我功能与用户登录持久化

1.先决条件&#xff1a;拿到了后端发的凭证并做了持久化储存 2.在1的基础上&#xff0c;加上一个记住我功能&#xff0c;记住我的体现暂时定为&#xff1a; a.不勾选记住我&#xff1a;在浏览器的对话窗不关闭的情况下&#xff0c;凭证是存在有效的&#xff0c;但关闭了对话框…...

SQL自学,mysql从入门到精通 --- 第 15天,数据导入、导出

数据的导入、导出 -- 查看当前设置的目录路径&#xff0c;限制从数据库服务器读取和写入文件的操作只能在指定的目录中进行,在安全性和文件操作限制方面具有重要意义。rootmysqldb 14:19: [(none)]> SHOW VARIABLES LIKE "secure_file_priv"; -----------------…...

android skia渲染介绍

Android AOSP 的渲染系统主要使用 Skia 图形库。Skia 是一个开源的 2D 图形库&#xff0c;它被广泛应用于 Android 的图形渲染中&#xff0c;负责绘制 UI 元素、文本、图像以及其他 2D 图形内容。 以下是 Android AOSP 中 Skia 的作用和它在渲染系统中的位置&#xff1a; 1. 什…...

【网络安全】服务器安装Docker及拉取镜像教程

文章目录 1. 安装 Docker2. 拉取镜像3. 运行 Ubuntu 容器4. 执行相关操作5. 退出并停止容器1. 安装 Docker # 更新软件包索引 sudo apt update# 安装必要的依赖 sudo apt install -y ca-certificates curl gnupg...

Day87:游戏事件绑定

在游戏开发中,事件绑定是指通过监听和处理用户的输入或其他事件(如鼠标点击、键盘按键、碰撞等),来控制游戏中的行为和流程。事件绑定在游戏中扮演着至关重要的角色,它能够让游戏具备互动性和实时反馈。 今天,我们将学习如何在 Python 中使用 Pygame 进行游戏事件绑定,…...

elementplus 使用日期时间选择器,设置可选范围为前后大于2年且只能选择历史时间不能大于当前时间点

需求&#xff1a;时间选择器可选的时间范围进行限制&#xff0c;-2年<a<2年且a<new Date().getTime()核心&#xff1a;这里需要注意plus版没有picker-options换成disabled-date属性了&#xff0c;使用了visible-change和calendar-change属性逻辑&#xff1a;另设一个参…...

将 AMD Zynq™ RFSoC 扩展到毫米波领域

目录 将 AMD Zynq™ RFSoC 扩展到毫米波领域Avnet XRF RFSoC 系统级模块适用于 MATLAB 的 Avnet RFSoC Explorer 工具箱5G mmWave PAAM 开发平台突破性的宽带毫米波波束成形特征&#xff1a;OTBF103 Mathworks Simulink 模型优化毫米波应用中的射频信号路径 用于宽带毫米波上/下…...

Redis企业开发实战(五)——点评项目之分布式锁Redission与秒杀优化

目录 一、Redisson (一)Redisson基本介绍 (二)Redisson入门 1.引入依赖 2.配置Redisson客户端 3.使用Redission的分布式锁 4.tryLock参数解析 4.1tryLock() 4.2tryLock(long waitTime, TimeUnit unit) 4.3tryLock(long waitTime, long leaseTime, TimeUnit unit) 4…...

IDEA安装离线插件(目前提供了MavenHelper安装包)

目录 1、离线安装方式2、Maven Helper 1、离线安装方式 首先访问 IDEA插件网站 下载离线插件安装包&#xff0c;操作如下&#xff1a; 然后打开IDEA的Settings配置&#xff0c;点击Plugins&#xff0c;点击右侧设置按钮&#xff08;齿轮&#xff09;&#xff0c;选择Install P…...

LabVIEW 开发航天项目软件

在航天项目软件开发中&#xff0c;LabVIEW 凭借其图形化编程优势被广泛应用。然而&#xff0c;航天项目的高可靠性、高精度及复杂环境适应性要求&#xff0c;使得在使用 LabVIEW 开发时&#xff0c;有诸多关键要点需要特别关注。本文将详细分析在开发航天项目软件时需要重点注意…...

互联网大厂中面试的高频计算机网络问题及详解

前言 哈喽各位小伙伴们,本期小梁给大家带来了互联网大厂中计算机网络部分的高频面试题,本文会以通俗易懂的语言以及图解形式描述,希望能给大家的面试带来一点帮助,祝大家offer拿到手软!!! 话不多说,我们立刻进入本期正题! 一、计算机网络基础部分 1 先来说说计算机网…...

如何定义“破坏环境”

当我们谈论破坏环境时&#xff0c;通常会从人类活动对自然生态造成负面影响的角度来定义。例如&#xff0c;大规模的森林砍伐、工业污染排放、温室气体增加等&#xff0c;都是典型的破坏环境的行为。我们常常看到这些行为导致了生态系统的破坏、物种灭绝、气候变化等问题&#…...

WPS接入DeepSeek模型

1.wps 下载安装 WPS-支持多人在线协作编辑Word、Excel和PPT文档_WPS官方网站 &#xff08;最好是安装最新的wps&#xff09; 2.offieceAi工具下载安装 软件下载 | OfficeAI助手 下载后安装下载下来的两个工具。安装路径可以自行修改 3.打开WPS,点击文件-》 选项-》信任中心 勾…...

自然语言处理NLP_[1]-NLP入门

文章目录 1.自然语言处理入门1. 什么是自然语言处理2.自然语言处理的发展简史3 自然语言处理的应用场景1. **机器翻译**2. **文本分类**3. **情感分析**4. **问答系统**5. **文本生成**6. **信息抽取**7. **语音识别与合成**8. **文本摘要**9. **搜索引擎优化**10. **聊天机器人…...