GeneratorIdUtils.java 3.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129
  1. package com.example.util;
  2. import java.net.InetAddress;
  3. import java.net.UnknownHostException;
  4. import java.text.SimpleDateFormat;
  5. /**
  6. * 订购业务唯一订单号实现
  7. *
  8. * @author ouyangjun
  9. */
  10. public class GeneratorIdUtils {
  11. /**
  12. * 用ip地址最后几个字节标示
  13. */
  14. private long workerId;
  15. /**
  16. * 可配置在properties中,启动时加载,此处默认先写成0
  17. */
  18. private long datacenterId = 0L;
  19. private long sequence = 0L;
  20. /**
  21. * 节点ID长度
  22. */
  23. private long workerIdBits = 8L;
  24. /**
  25. * 序列号12位
  26. */
  27. private long sequenceBits = 12L;
  28. /**
  29. * 机器节点左移12位
  30. */
  31. private long workerIdShift = sequenceBits;
  32. /**
  33. * 数据中心节点左移17位
  34. */
  35. private long datacenterIdShift = sequenceBits + workerIdBits;
  36. /**
  37. * 4095
  38. */
  39. private long sequenceMask = -1L ^ (-1L << sequenceBits);
  40. private long lastTimestamp = -1L;
  41. public GeneratorIdUtils() {
  42. workerId = 0x000000FF & getLastIP();
  43. }
  44. /**
  45. * 调用该方法,获取序列ID
  46. *
  47. * @return
  48. */
  49. public synchronized String nextId() {
  50. //获取当前毫秒数
  51. long timestamp = currentTime();
  52. //如果服务器时间有问题(时钟后退) 报错。
  53. if (timestamp < lastTimestamp) {
  54. throw new RuntimeException(String.format("时钟向后移动。拒绝在%d毫秒内生成id", lastTimestamp - timestamp));
  55. }
  56. //如果上次生成时间和当前时间相同,在同一毫秒内
  57. if (lastTimestamp == timestamp) {
  58. //sequence自增,因为sequence只有12bit,所以和sequenceMask相与一下,去掉高位
  59. sequence = (sequence + 1) & sequenceMask;
  60. //判断是否溢出,也就是每毫秒内超过4095,当为4096时,与sequenceMask相与,sequence就等于0
  61. if (sequence == 0) {
  62. //自旋等待到下一毫秒
  63. timestamp = nextMillis(lastTimestamp);
  64. }
  65. } else {
  66. //如果和上次生成时间不同,重置sequence,就是下一毫秒开始,sequence计数重新从0开始累加
  67. sequence = 0L;
  68. }
  69. lastTimestamp = timestamp;
  70. long suffix = (datacenterId << datacenterIdShift) | (workerId << workerIdShift) | sequence;
  71. // 格式化日期
  72. SimpleDateFormat timePe = new SimpleDateFormat("yyyyMMddHHMMssSSS");
  73. String datePrefix = timePe.format(timestamp);
  74. return datePrefix + suffix;
  75. }
  76. /**
  77. * 比较当前时间戳和下一个时间戳,如果下一个时间戳等于或小于当前时间戳,则循环获取下个时间戳
  78. * 该方法主要是避免同一时间获取同一时间戳
  79. *
  80. * @param lastTimestamp
  81. * @return
  82. */
  83. protected long nextMillis(long lastTimestamp) {
  84. long timestamp = currentTime();
  85. while (timestamp <= lastTimestamp) {
  86. timestamp = currentTime();
  87. }
  88. return timestamp;
  89. }
  90. /**
  91. * 获取系统当前时间戳
  92. *
  93. * @return
  94. */
  95. protected long currentTime() {
  96. return System.currentTimeMillis();
  97. }
  98. /**
  99. * 获取当前本地IP
  100. *
  101. * @return
  102. */
  103. private byte getLastIP() {
  104. byte lastip = 0;
  105. try {
  106. InetAddress ip = InetAddress.getLocalHost();
  107. byte[] ipByte = ip.getAddress();
  108. lastip = ipByte[ipByte.length - 1];
  109. } catch (UnknownHostException e) {
  110. e.printStackTrace();
  111. }
  112. return lastip;
  113. }
  114. }