123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110 |
- using System;
- using System.Collections.Generic;
- using System.Linq;
- using System.Text;
- using System.Threading.Tasks;
- using YiSha.Util;
- namespace YiSha.IdGenerator
- {
- public class Snowflake
- {
- private const long TwEpoch = 1546272000000L;//2019-01-01 00:00:00
- 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 long _sequence = 0L;
- private long _lastTimestamp = -1L;
- /// <summary>
- ///10位的数据机器位中的高位
- /// </summary>
- public long WorkerId { get; protected set; }
- /// <summary>
- /// 10位的数据机器位中的低位
- /// </summary>
- public long DatacenterId { get; protected set; }
- private readonly object _lock = new object();
- /// <summary>
- /// 基于Twitter的snowflake算法
- /// </summary>
- /// <param name="workerId">10位的数据机器位中的高位,默认不应该超过5位(5byte)</param>
- /// <param name="datacenterId"> 10位的数据机器位中的低位,默认不应该超过5位(5byte)</param>
- /// <param name="sequence">初始序列</param>
- public Snowflake(long workerId, long datacenterId, long sequence = 0L)
- {
- WorkerId = workerId;
- DatacenterId = datacenterId;
- _sequence = sequence;
- 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");
- }
- }
- public long CurrentId { get; private set; }
- /// <summary>
- /// 获取下一个Id,该方法线程安全
- /// </summary>
- /// <returns></returns>
- public long NextId()
- {
- lock (_lock)
- {
- var timestamp = DateTimeHelper.GetUnixTimeStamp(DateTime.Now);
- if (timestamp < _lastTimestamp)
- {
- //TODO 是否可以考虑直接等待?
- throw new Exception(
- $"Clock moved backwards or wrapped around. Refusing to generate id for {_lastTimestamp - timestamp} ticks");
- }
- if (_lastTimestamp == timestamp)
- {
- _sequence = (_sequence + 1) & SequenceMask;
- if (_sequence == 0)
- {
- timestamp = TilNextMillis(_lastTimestamp);
- }
- }
- else
- {
- _sequence = 0;
- }
- _lastTimestamp = timestamp;
- CurrentId = ((timestamp - TwEpoch) << TimestampLeftShift) |
- (DatacenterId << DatacenterIdShift) |
- (WorkerId << WorkerIdShift) | _sequence;
- return CurrentId;
- }
- }
- private long TilNextMillis(long lastTimestamp)
- {
- var timestamp = DateTimeHelper.GetUnixTimeStamp(DateTime.Now);
- while (timestamp <= lastTimestamp)
- {
- timestamp = DateTimeHelper.GetUnixTimeStamp(DateTime.Now);
- }
- return timestamp;
- }
- }
- }
|