SnowFlake.cs 9.0 KB

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