legongju.com
我们一直在努力
2025-01-12 23:26 | 星期天

C# Snowflake算法的源码解析

Snowflake 算法是 Twitter 开源的一种分布式 ID 生成策略,它可以在不依赖数据库或其他存储设备的情况下生成全局唯一的 ID。Snowflake 算法的 ID 结构包括时间戳、数据中心 ID、机器 ID 和序列号等部分。

以下是一个简单的 C# 实现 Snowflake 算法的示例代码:

using System;
using System.Threading;

public class Snowflake
{
    private const long Twepoch = 1288834974657L;
    private const int WorkerIdBits = 5;
    private const int DatacenterIdBits = 5;
    private const int SequenceBits = 12;
    private const long MaxWorkerId = -1L ^ (-1L<< WorkerIdBits);
    private const long MaxDatacenterId = -1L ^ (-1L<< DatacenterIdBits);
    private const int WorkerIdShift = SequenceBits;
    private const int DatacenterIdShift = SequenceBits + WorkerIdBits;
    private const int TimestampLeftShift = SequenceBits + WorkerIdBits + DatacenterIdBits;
    private const long SequenceMask = -1L ^ (-1L << SequenceBits);

    private static long _sequence = 0L;
    private static long _lastTimestamp = -1L;

    private readonly long _workerId;
    private readonly long _datacenterId;
    private readonly object _lock = new object();

    public Snowflake(long workerId, long datacenterId)
    {
        if (workerId > MaxWorkerId || workerId < 0)
        {
            throw new ArgumentException($"Worker Id can't be greater than {MaxWorkerId} or less than 0");
        }

        if (datacenterId > MaxDatacenterId || datacenterId < 0)
        {
            throw new ArgumentException($"Datacenter Id can't be greater than {MaxDatacenterId} or less than 0");
        }

        _workerId = workerId;
        _datacenterId = datacenterId;
    }

    public long NextId()
    {
        lock (_lock)
        {
            var timestamp = GetCurrentTimestamp();

            if (timestamp > _lastTimestamp)
            {
                _sequence = 0;
                _lastTimestamp = timestamp;
            }
            else
            {
                _sequence = (_sequence + 1) & SequenceMask;

                if (_sequence == 0)
                {
                    timestamp = WaitNextMillis(_lastTimestamp);
                    _lastTimestamp = timestamp;
                }
            }

            return ((timestamp - Twepoch)<< TimestampLeftShift) |
                   (_datacenterId<< DatacenterIdShift) |
                   (_workerId<< WorkerIdShift) |
                   _sequence;
        }
    }

    private long GetCurrentTimestamp()
    {
        return (long)(DateTime.UtcNow - new DateTime(1970, 1, 1, 0, 0, 0, DateTimeKind.Utc)).TotalMilliseconds;
    }

    private long WaitNextMillis(long lastTimestamp)
    {
        var timestamp = GetCurrentTimestamp();

        while (timestamp <= lastTimestamp)
        {
            Thread.Sleep(1);
            timestamp = GetCurrentTimestamp();
        }

        return timestamp;
    }
}

这个实现中,我们定义了一个 Snowflake 类,它包含了 TwepochWorkerIdBitsDatacenterIdBitsSequenceBits 等常量,用于计算 ID 的各个部分。同时,我们还定义了一些私有变量,如 _sequence_lastTimestamp_workerId_datacenterId,用于存储当前的序列号、最后一次生成 ID 的时间戳、工作节点 ID 和数据中心 ID。

Snowflake 类的构造函数接收两个参数,分别是工作节点 ID 和数据中心 ID,并进行合法性检查。NextId 方法用于生成下一个 ID,它首先获取当前的时间戳,然后根据时间戳、工作节点 ID、数据中心 ID 和序列号计算出一个新的 ID。如果当前时间戳小于上一次生成 ID 的时间戳,说明系统时钟回拨,此时需要等待下一毫秒再生成 ID。

GetCurrentTimestamp 方法用于获取当前的时间戳(毫秒级),WaitNextMillis 方法用于等待下一毫秒。

这个实现是线程安全的,因为我们使用了 lock 关键字来确保在生成 ID 时不会被其他线程打断。

未经允许不得转载 » 本文链接:https://www.legongju.com/article/104218.html

相关推荐

  • c++ hashmap用法是什么

    c++ hashmap用法是什么

    在C++中,hash_map 是一个非标准的容器,它提供了基于哈希表的键值对存储。然而,需要注意的是,自从C++11起,hash_map已经被废弃,并在C++17中被移除。取而代之...

  • C#中token是什么及其作用

    C#中token是什么及其作用

    在C#中,token通常指的是一个代表特定字符串或数据的标识符 身份验证和授权:在许多Web应用程序和API中,使用令牌(如JWT,JSON Web Token)进行身份验证和授权。...

  • 在C#中如何对token进行加密

    在C#中如何对token进行加密

    在C#中,对token进行加密通常使用的是对称加密算法,如AES(Advanced Encryption Standard)。以下是一个使用AES加密和解密token的示例:
    首先,需要引入Sy...

  • C#中token传递方式有哪些

    C#中token传递方式有哪些

    在C#中,token传递方式主要有以下几种: 查询字符串(Query String):将token作为URL的一部分,通过查询字符串传递。这种方式简单易用,但不太安全,因为URL可能...

  • Snowflake与C#的集成步骤

    Snowflake与C#的集成步骤

    要将Snowflake与C#集成,你需要遵循以下步骤: 安装Snowflake C#驱动程序:首先,你需要在你的项目中安装Snowflake C#驱动程序。你可以通过NuGet包管理器或者使用...

  • C# Snowflake算法的适用范围

    C# Snowflake算法的适用范围

    Snowflake 算法是一种分布式 ID 生成策略,主要用于在分布式系统中生成唯一、有序、不重复的 ID。它可以广泛应用于各种需要全局唯一 ID 的场景,例如数据库记录、...

  • 如何在C#项目中引入Snowflake

    如何在C#项目中引入Snowflake

    要在C#项目中引入Snowflake,您需要首先安装Snowflake的.NET驱动程序,然后使用它来连接到Snowflake数据库 安装Snowflake .NET驱动程序:
    打开NuGet包管理器...

  • Snowflake算法原理及C#实现

    Snowflake算法原理及C#实现

    Snowflake 是 Twitter 开源的一个分布式 ID 生成算法,用于在分布式系统中生成唯一、有序、不重复的 ID。它的设计目标是在不依赖数据库或其他存储设备的情况下生...