SnowFlake.cs 8.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256
  1. using Amazon.Runtime.Internal.Util;
  2. using System.Net;
  3. using System.Runtime.InteropServices;
  4. namespace Business.Core.Utilities
  5. {
  6. /// <summary>
  7. /// 雪花算法生成id
  8. /// </summary>
  9. public class SnowFlake
  10. {
  11. /// <summary>
  12. /// 机器id所占的位数
  13. /// </summary>
  14. private const int workerIdBits = 5;
  15. /// <summary>
  16. /// 数据表述id所占位数
  17. /// </summary>
  18. private const int datacenterIdBits = 5;
  19. /// <summary>
  20. /// 支持的最大机器id,结果是31
  21. /// </summary>
  22. private const long maxWorkerId = -1L ^ (-1L << workerIdBits);
  23. /// <summary>
  24. /// 支持的最大数据表示id,结果是31
  25. /// </summary>
  26. private const long maxDatacenterId = -1L ^ (-1L << datacenterIdBits);
  27. /// <summary>
  28. /// 序列在id中所占的位数
  29. /// </summary>
  30. private const int sequenceBits = 12;
  31. /// <summary>
  32. /// 数据标识id向左移17位
  33. /// </summary>
  34. private const int datacenterIdShift = sequenceBits + workerIdBits;
  35. /// <summary>
  36. /// 机器ID向左移12位
  37. /// </summary>
  38. private const int workerIdShift = sequenceBits;
  39. private const int timestampLeftShift = sequenceBits + workerIdBits + datacenterIdBits;
  40. /// <summary>
  41. /// 生成序列的掩码
  42. /// </summary>
  43. private const long sequenceMask = -1L ^ (-1L << sequenceBits);
  44. /// <summary>
  45. /// 数据中心ID(0-31)
  46. /// </summary>
  47. public long DatacenterId { get; private set; }
  48. /// <summary>
  49. /// 工作机器ID(0-31)
  50. /// </summary>
  51. public long WorkerId { get; private set; }
  52. /// <summary>
  53. /// 毫秒内序列(0-4095)
  54. /// </summary>
  55. public long Sequence { get; private set; }
  56. /// <summary>
  57. /// 上次生成ID的时间戳
  58. /// </summary>
  59. public long LastTimestamp { get; private set; }
  60. /// <summary>
  61. /// 开始时间戳。首次使用前设置,否则无效,默认2010-1-1
  62. /// </summary>
  63. public DateTime StartTimestamp { get; set; } = new DateTime(2010,1,1);
  64. static object syncRoot = new object();
  65. static readonly Lazy<SnowFlake> snowflake = new(() => new SnowFlake());
  66. /// <summary>
  67. /// 默认静态实例,WorkerId = 0,DatacenterId = 0
  68. /// </summary>
  69. public static SnowFlake Instance { get; } = snowflake.Value;
  70. /// <summary>
  71. /// 构造函数
  72. /// </summary>
  73. /// <exception cref="ArgumentOutOfRangeException"></exception>
  74. public SnowFlake()
  75. {
  76. //生成workerid和datacenterid
  77. SetWorkIdAndDatacenterId();
  78. if (WorkerId > maxWorkerId || WorkerId < 0)
  79. {
  80. throw new ArgumentOutOfRangeException(nameof(WorkerId),$"不能大于{maxWorkerId}或小于0");
  81. }
  82. if (DatacenterId > maxDatacenterId || DatacenterId < 0)
  83. {
  84. throw new ArgumentOutOfRangeException(nameof(DatacenterId), $"不能大于{maxDatacenterId}或小于0");
  85. }
  86. Sequence = 0L;
  87. LastTimestamp = -1L;
  88. }
  89. /// <summary>
  90. /// 生成WorkerId和DatacenterId
  91. /// </summary>
  92. /// <returns></returns>
  93. private void SetWorkIdAndDatacenterId()
  94. {
  95. string? ipAdress = "";
  96. try
  97. {
  98. string? hostName;
  99. if (RuntimeInformation.IsOSPlatform(OSPlatform.Linux))
  100. {
  101. hostName = Environment.GetEnvironmentVariable("MY_NODE_NAME");
  102. ipAdress = Environment.GetEnvironmentVariable("MY_HOST_IP");
  103. }
  104. else
  105. {
  106. hostName = Dns.GetHostName();
  107. IPHostEntry ipEntity = Dns.GetHostEntry(hostName);
  108. for (int i = 0; i < ipEntity.AddressList.Length; i++)
  109. {
  110. if (ipEntity.AddressList[i].AddressFamily == System.Net.Sockets.AddressFamily.InterNetwork)
  111. {
  112. ipAdress = ipEntity.AddressList[i].ToString();
  113. }
  114. }
  115. }
  116. if (string.IsNullOrEmpty(ipAdress) || string.IsNullOrEmpty(hostName))
  117. {
  118. WorkerId = 0;
  119. DatacenterId = 0;
  120. }
  121. else
  122. {
  123. WorkerId = StringToLong(ipAdress);
  124. DatacenterId = StringToLong(hostName);
  125. }
  126. }
  127. catch (Exception)
  128. {
  129. WorkerId = 0;
  130. DatacenterId = 0;
  131. }
  132. }
  133. /// <summary>
  134. /// 生成workerid和datacenterid
  135. /// </summary>
  136. /// <param name=""></param>
  137. /// <returns></returns>
  138. private long StringToLong(string strInput)
  139. {
  140. int[] arrays = ToCodePoints(strInput);
  141. int sum = 0;
  142. for (int i = 0; i < arrays.Length; i++)
  143. {
  144. sum += arrays[i];
  145. }
  146. return (long)(sum % 32);
  147. }
  148. /// <summary>
  149. /// 将字符串转换成数组
  150. /// </summary>
  151. /// <param name="strInput"></param>
  152. /// <returns></returns>
  153. private int[] ToCodePoints(string strInput)
  154. {
  155. var codePoints = new List<int>(strInput.Length);
  156. for (int i = 0; i < strInput.Length; i++)
  157. {
  158. codePoints.Add(char.ConvertToUtf32(strInput, i));
  159. if (char.IsHighSurrogate(strInput[i]))
  160. {
  161. i += 1;
  162. }
  163. }
  164. return codePoints.ToArray();
  165. }
  166. /// <summary>
  167. /// 获取下一个ID
  168. /// </summary>
  169. /// <returns></returns>
  170. public long NextId()
  171. {
  172. lock (syncRoot)
  173. {
  174. //获取当前时间戳
  175. long timestamp = GetCurrentTimestamp();
  176. if (timestamp > LastTimestamp)//时间戳改变,毫秒内序列重置
  177. {
  178. Sequence = 0L;
  179. }
  180. else if (timestamp == LastTimestamp)//如果是同一时间生成的,则进行毫秒内序列
  181. {
  182. Sequence = (Sequence + 1) & sequenceMask;
  183. if (Sequence == 0)//毫秒内序列溢出
  184. {
  185. timestamp = GetNextTimestamp(LastTimestamp);//阻塞到下一个毫秒,获取新的时间戳
  186. }
  187. }
  188. else//当前时间小于上一次ID生成的时间戳,证明系统时钟被回拨,此时需要做回拨处理
  189. {
  190. Sequence = (Sequence + 1) & sequenceMask;
  191. if (Sequence > 0)
  192. {
  193. timestamp = LastTimestamp;//停留再最后一次时间戳上,等待系统时间追上后即完全度过了时钟回拨问题
  194. }
  195. else {
  196. timestamp = LastTimestamp + 1;//直接进位到下一个毫秒
  197. }
  198. }
  199. LastTimestamp = timestamp;//上次生成ID的时间戳
  200. //移动并通过或运算拼到一起组成64位的ID
  201. var id = (timestamp << timestampLeftShift)
  202. | (DatacenterId << datacenterIdShift)
  203. | (WorkerId << workerIdShift)
  204. | Sequence;
  205. return id;
  206. }
  207. }
  208. /// <summary>
  209. /// 阻塞到下一个毫秒,直到获得新的时间戳
  210. /// </summary>
  211. /// <param name="lastTimestamp"></param>
  212. /// <returns></returns>
  213. private long GetNextTimestamp(long lastTimestamp)
  214. {
  215. //获取当前时间戳
  216. long timestamp = GetCurrentTimestamp();
  217. while (timestamp <= lastTimestamp)
  218. {
  219. timestamp = GetCurrentTimestamp();
  220. }
  221. return timestamp;
  222. }
  223. /// <summary>
  224. /// 获取当前时间戳
  225. /// </summary>
  226. /// <returns></returns>
  227. private long GetCurrentTimestamp()
  228. {
  229. return (long)(DateTime.Now - StartTimestamp).TotalMilliseconds;
  230. }
  231. }
  232. }