雪花算法应用

news2025/2/11 8:37:53

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

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

package snowflake

import (
    "sync"
    "time"
    "errors"
)

// Snowflake 结构体
type Snowflake struct {
    mutex         sync.Mutex    // 互斥锁
    lastTimestamp int64        // 上次的时间戳
    workerId      int64        // 工作节点ID
    datacenterId  int64        // 数据中心ID
    sequence      int64        // 序列号
}

const (
    workerBits     = uint(5)                              // 工作节点ID位数
    datacenterBits = uint(5)                              // 数据中心ID位数
    sequenceBits   = uint(12)                             // 序列号位数
    
    maxWorkerId     = -1 ^ (-1 << workerBits)            // 最大工作节点ID
    maxDatacenterId = -1 ^ (-1 << datacenterBits)        // 最大数据中心ID
    maxSequence     = -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

    // 组合ID
    id := ((timestamp - epoch) << timeShift) |
          (s.datacenterId << datacenterShift) |
          (s.workerId << workerShift) |
          s.sequence

    return id, nil
}
package main

import (
    "fmt"
    "log"
    "your/path/snowflake"
)

func main() {
    // 创建雪花算法实例(数据中心ID=1, 工作节点ID=1)
    sf, err := snowflake.NewSnowflake(1, 1)
    if err != nil {
        log.Fatal(err)
    }

    // 生成ID
    for 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.Mutex
    lastTimestamp int64
    workerId      int64
    datacenterId  int64
    sequence      int64
    maxWaitTime   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 - timestamp
        if 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.Mutex
    lastTimestamp int64
    workerId      int64
    datacenterId  int64
    sequence      int64
    backupDelta   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) & maxSequence
        if s.sequence == 0 {
            timestamp = s.waitNextMillis(timestamp)
        }
    } else {
        s.sequence = 0
    }

    s.lastTimestamp = timestamp
    // ... 生成ID的逻辑
}

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

type SegmentSnowflake struct {
    mutex         sync.Mutex
    lastTimestamp int64
    workerId      int64
    datacenterId  int64
    sequence      int64
    currentSegment int64
    segmentBits   uint
    segmentSize   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 = 0
        return s.generateId(s.lastTimestamp, s.currentSegment, s.sequence), nil
    }

    if timestamp != s.lastTimestamp {
        s.currentSegment = 0
        s.sequence = 0
    }

    s.lastTimestamp = timestamp
    return 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. 在实际生产环境中,建议结合多种方案,并加入监控和告警机制
选择哪种方案需要根据具体的业务场景来决定:

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

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/2296226.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

Chapter3:结构化程序设计

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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 先来说说计算机网…

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. **聊天机器人…

计算机毕业设计Python+Spark知识图谱医生推荐系统 医生门诊预测系统 医生数据分析 医生可视化 医疗数据分析 医生爬虫 大数据毕业设计 机器学习

温馨提示&#xff1a;文末有 CSDN 平台官方提供的学长联系方式的名片&#xff01; 温馨提示&#xff1a;文末有 CSDN 平台官方提供的学长联系方式的名片&#xff01; 温馨提示&#xff1a;文末有 CSDN 平台官方提供的学长联系方式的名片&#xff01; 作者简介&#xff1a;Java领…

Selenium常用自动化函数

博主主页: 码农派大星. 数据结构专栏:Java数据结构 数据库专栏:数据库 JavaEE专栏:JavaEE 软件测试专栏:软件测试 关注博主带你了解更多知识 目录 1.元素的定位 1.1 定位步骤 1,要想定位,就先打开开发者工具 2,先点击左上角图标 1.2 cssSelector 1.3 xpath 2.操作测…

【故障排除】ls: command not found 终端命令失效的解决办法

【TroubleShooting】ls: command not found 终端命令失效的解决办法 A Solution to Solve “Command not found” of Terminal on Mac 一直在使用心爱的MacBook Pro的Terminal&#xff0c;并且为她定制了不同的Profile。 这样&#xff0c;看起来她可以在不同季节&#xff0c…

12.翻转、对称二叉树,二叉树的深度

反转二叉树 递归写法 很简单 class Solution { public:TreeNode* invertTree(TreeNode* root) {if(rootnullptr)return root;TreeNode* tmp;tmproot->left;root->leftroot->right;root->righttmp;invertTree(root->left);invertTree(root->right);return …