代码实现

使用一个 64 bit 的 long 型的数字作为全局唯一 id。在分布式系统中的应用十分广泛,且ID 引入了时间戳。基本上保持自增的,后面的代码中有详细的注解。

  1. class IdWorker {
  2. //因为二进制里第一个 bit 为如果是 1,那么都是负数,但是我们生成的 id 都是正数,所以第一个 bit 统一都是 0。
  3. //机器ID 2进制5位 32位减掉1位 31个
  4. private long workerId;
  5. //机房ID 2进制5位 32位减掉1位 31个
  6. private long datacenterId;
  7. //代表一毫秒内生成的多个id的最新序号 12位 4096 -1 = 4095 个
  8. private long sequence;
  9. //设置一个时间初始值 2^41 - 1 差不多可以用69年
  10. private long twepoch = 1585644268888L;
  11. //5位的机器id
  12. private long workerIdBits = 5L;
  13. //5位的机房id
  14. private long datacenterIdBits = 5L;
  15. //每毫秒内产生的id数 2 的 12次方
  16. private long sequenceBits = 12L;
  17. // 这个是二进制运算,就是5 bit最多只能有31个数字,也就是说机器id最多只能是32以内
  18. private long maxWorkerId = -1L ^ (-1L << workerIdBits);
  19. // 这个是一个意思,就是5 bit最多只能有31个数字,机房id最多只能是32以内
  20. private long maxDatacenterId = -1L ^ (-1L << datacenterIdBits);
  21. private long workerIdShift = sequenceBits;
  22. private long datacenterIdShift = sequenceBits + workerIdBits;
  23. private long timestampLeftShift = sequenceBits + workerIdBits + datacenterIdBits;
  24. private long sequenceMask = -1L ^ (-1L << sequenceBits);
  25. //记录产生时间毫秒数,判断是否是同1毫秒
  26. private long lastTimestamp = -1L;
  27. public IdWorker(long workerId, long datacenterId, long sequence) {
  28. // 检查机房id和机器id是否超过31 不能小于0
  29. if (workerId > maxWorkerId || workerId < 0) {
  30. throw new IllegalArgumentException(
  31. String.format("worker Id can't be greater than %d or less than 0",maxWorkerId));
  32. }
  33. if (datacenterId > maxDatacenterId || datacenterId < 0) {
  34. throw new IllegalArgumentException(
  35. String.format("datacenter Id can't be greater than %d or less than 0",maxDatacenterId));
  36. }
  37. this.workerId = workerId;
  38. this.datacenterId = datacenterId;
  39. this.sequence = sequence;
  40. }
  41. /**
  42. * main 测试类
  43. * @param args
  44. */
  45. public static void main(String[] args) {
  46. System.out.println(1&4596);
  47. System.out.println(2&4596);
  48. System.out.println(6&4596);
  49. System.out.println(6&4596);
  50. System.out.println(6&4596);
  51. System.out.println(6&4596);
  52. IdWorker worker = new IdWorker(1,1,1);
  53. for (int i = 0; i < 22; i++) {
  54. System.out.println(worker.nextId());
  55. }
  56. }
  57. public long getWorkerId() {
  58. return workerId;
  59. }
  60. public long getDatacenterId() {
  61. return datacenterId;
  62. }
  63. public long getTimestamp() {
  64. return System.currentTimeMillis();
  65. }
  66. // 这个是核心方法,通过调用nextId()方法,让当前这台机器上的snowflake算法程序生成一个全局唯一的id
  67. public synchronized long nextId() {
  68. // 这儿就是获取当前时间戳,单位是毫秒
  69. long timestamp = timeGen();
  70. if (timestamp < lastTimestamp) {
  71. System.err.printf("clock is moving backwards. Rejecting requests until %d.", lastTimestamp);
  72. throw new RuntimeException(String.format(
  73. "Clock moved backwards. Refusing to generate id for %d milliseconds", lastTimestamp - timestamp));
  74. }
  75. // 下面是说假设在同一个毫秒内,又发送了一个请求生成一个id
  76. // 这个时候就得把seqence序号给递增1,最多就是4096
  77. if (lastTimestamp == timestamp) {
  78. // 这个意思是说一个毫秒内最多只能有4096个数字,无论你传递多少进来,
  79. //这个位运算保证始终就是在4096这个范围内,避免你自己传递个sequence超过了4096这个范围
  80. sequence = (sequence + 1) & sequenceMask;
  81. //当某一毫秒的时间,产生的id数 超过4095,系统会进入等待,直到下一毫秒,系统继续产生ID
  82. if (sequence == 0) {
  83. timestamp = tilNextMillis(lastTimestamp);
  84. }
  85. } else {
  86. sequence = 0;
  87. }
  88. // 这儿记录一下最近一次生成id的时间戳,单位是毫秒
  89. lastTimestamp = timestamp;
  90. // 这儿就是最核心的二进制位运算操作,生成一个64bit的id
  91. // 先将当前时间戳左移,放到41 bit那儿;将机房id左移放到5 bit那儿;将机器id左移放到5 bit那儿;将序号放最后12 bit
  92. // 最后拼接起来成一个64 bit的二进制数字,转换成10进制就是个long型
  93. return ((timestamp - twepoch) << timestampLeftShift) | (datacenterId << datacenterIdShift)
  94. | (workerId << workerIdShift) | sequence;
  95. }
  96. /**
  97. * 当某一毫秒的时间,产生的id数 超过4095,系统会进入等待,直到下一毫秒,系统继续产生ID
  98. * @param lastTimestamp
  99. * @return
  100. */
  101. private long tilNextMillis(long lastTimestamp) {
  102. long timestamp = timeGen();
  103. while (timestamp <= lastTimestamp) {
  104. timestamp = timeGen();
  105. }
  106. return timestamp;
  107. }
  108. //获取当前时间戳
  109. private long timeGen() {
  110. return System.currentTimeMillis();
  111. }
  112. }

缺点是什么?

  1. 时间回拨导致ID重复。
  2. 41位时间戳只能用70年,所以尽量不要用1970年作为时间起始点,而用应用发布时间可以更好扩展阈值。
  3. 周期内消耗完毕需要等下下一周期(最大可达4百万的tps,2^12*1000)。