SnowflakeIdGenerator-雪花算法id生成方法

简介: SnowflakeIdGenerator-雪花算法id生成方法

package com.todod.common.util;


import java.util.Date;


/**
* Twitter_Snowflake<br>
* SnowFlake的结构如下(每部分用-分开):<br>
* 0 - 0000000000 0000000000 0000000000 0000000000 0 - 00000 - 00000 -
* 000000000000 <br>
* 1位标识,由于long基本类型在Java中是带符号的,最高位是符号位,正数是0,负数是1,所以id一般是正数,最高位是0<br>
* 41位时间截(毫秒级),注意,41位时间截不是存储当前时间的时间截,而是存储时间截的差值(当前时间截 - 开始时间截)
* 得到的值),这里的的开始时间截,一般是我们的id生成器开始使用的时间,由我们程序来指定的(如下下面程序IdWorker类的startTime属性)。41位的时间截,可以使用69年,年T
* = (1L << 41) / (1000L * 60 * 60 * 24 * 365) = 69<br>
* 10位的数据机器位,可以部署在1024个节点,包括5datacenterId5workerId<br>
* 12位序列,毫秒内的计数,12位的计数顺序号支持每个节点每毫秒(同一机器,同一时间截)产生4096ID序号<br>
* 加起来刚好64位,为一个Long型。<br>
* SnowFlake的优点是,整体上按照时间自增排序,并且整个分布式系统内不会产生ID碰撞(由数据中心ID和机器ID作区分),并且效率较高,经测试,SnowFlake每秒能够产生26ID左右。
*/
public class SnowflakeIdGenerator {

   private static SnowflakeIdWorker idWorker;


   static {

      idWorker = new SnowflakeIdWorker(0, 0);

   }


   public static Long getId() {

      return idWorker.nextId();

   }


   private static class SnowflakeIdWorker {


      // ==============================Fields===========================================
      /** 开始时间截 (2015-01-01) */
      private final long twepoch = 1420041600000L;


      /** 机器id所占的位数 */
      private final long workerIdBits = 5L;


      /** 数据标识id所占的位数 */
      private final long datacenterIdBits = 5L;


      /** 支持的最大机器id,结果是31 (这个移位算法可以很快的计算出几位二进制数所能表示的最大十进制数) */
      private final long maxWorkerId = -1L ^ (-1L << workerIdBits);


      /** 支持的最大数据标识id,结果是31 */
      private final long maxDatacenterId = -1L ^ (-1L << datacenterIdBits);


      /** 序列在id中占的位数 */
      private final long sequenceBits = 12L;


      /** 机器ID向左移12*/
      private final long workerIdShift = sequenceBits;


      /** 数据标识id向左移17(12+5) */
      private final long datacenterIdShift = sequenceBits + workerIdBits;


      /** 时间截向左移22(5+5+12) */
      private final long timestampLeftShift = sequenceBits + workerIdBits + datacenterIdBits;


      /** 生成序列的掩码,这里为4095 (0b111111111111=0xfff=4095) */
      private final long sequenceMask = -1L ^ (-1L << sequenceBits);


      /** 工作机器ID(0~31) */
      private long workerId;


      /** 数据中心ID(0~31) */
      private long datacenterId;


      /** 毫秒内序列(0~4095) */
      private long sequence = 0L;


      /** 上次生成ID的时间截 */
      private long lastTimestamp = new Date().getTime();


      // ==============================Constructors=====================================
      /**
       * 构造函数
       *
       * @param workerId     工作ID (0~31)
       * @param datacenterId 数据中心ID (0~31)
       */
      public SnowflakeIdWorker(long workerId, long datacenterId) {

         if (workerId > maxWorkerId || workerId < 0) {

            throw new IllegalArgumentException(

                  String.format("worker Id can't be greater than %d or less than 0", maxWorkerId));

         }

         if (datacenterId > maxDatacenterId || datacenterId < 0) {

            throw new IllegalArgumentException(

                  String.format("datacenter Id can't be greater than %d or less than 0", maxDatacenterId));

         }

         this.workerId = workerId;

         this.datacenterId = datacenterId;

      }


      // ==============================Methods==========================================
      /**
       * 获得下一个ID (该方法是线程安全的)
       *
       * @return SnowflakeId
       */
      public synchronized long nextId() {

         long timestamp = timeGen();


         // 如果当前时间小于上一次ID生成的时间戳,说明系统时钟回退过这个时候应当抛出异常
         if (timestamp < lastTimestamp) {

            throw new RuntimeException(

                  String.format("Clock moved backwards.  Refusing to generate id for %d milliseconds",

                        lastTimestamp - timestamp));

         }


         // 如果是同一时间生成的,则进行毫秒内序列
         if (lastTimestamp == timestamp) {

            sequence = (sequence + 1) & sequenceMask;

            // 毫秒内序列溢出
            if (sequence == 0) {

               // 阻塞到下一个毫秒,获得新的时间戳
               timestamp = tilNextMillis(lastTimestamp);

            }

         }

         // 时间戳改变,毫秒内序列重置
         else {

            sequence = 0L;

         }


         // 上次生成ID的时间截
         lastTimestamp = timestamp;


         // 移位并通过或运算拼到一起组成64位的ID
         return ((timestamp - twepoch) << timestampLeftShift) //
               | (datacenterId << datacenterIdShift) //
               | (workerId << workerIdShift) //
               | sequence;

      }


      /**
       * 阻塞到下一个毫秒,直到获得新的时间戳
       *
       * @param lastTimestamp 上次生成ID的时间截
       * @return 当前时间戳
       */
      protected long tilNextMillis(long lastTimestamp) {

         long timestamp = timeGen();

         while (timestamp <= lastTimestamp) {

            timestamp = timeGen();

         }

         return timestamp;

      }


      /**
       * 返回以毫秒为单位的当前时间
       *
       * @return 当前时间(毫秒)
       */
      protected long timeGen() {

         return System.currentTimeMillis();

      }


   }

}

目录
相关文章
|
2月前
|
机器学习/深度学习 算法 数据挖掘
K-means聚类算法是机器学习中常用的一种聚类方法,通过将数据集划分为K个簇来简化数据结构
K-means聚类算法是机器学习中常用的一种聚类方法,通过将数据集划分为K个簇来简化数据结构。本文介绍了K-means算法的基本原理,包括初始化、数据点分配与簇中心更新等步骤,以及如何在Python中实现该算法,最后讨论了其优缺点及应用场景。
116 4
|
3月前
|
存储 算法 Java
解析HashSet的工作原理,揭示Set如何利用哈希算法和equals()方法确保元素唯一性,并通过示例代码展示了其“无重复”特性的具体应用
在Java中,Set接口以其独特的“无重复”特性脱颖而出。本文通过解析HashSet的工作原理,揭示Set如何利用哈希算法和equals()方法确保元素唯一性,并通过示例代码展示了其“无重复”特性的具体应用。
61 3
|
5月前
|
算法 Go
[go 面试] 雪花算法与分布式ID生成
[go 面试] 雪花算法与分布式ID生成
|
2月前
|
算法 关系型数据库 MySQL
分布式唯一ID生成:深入理解Snowflake算法在Go中的实现
在分布式系统中,确保每个节点生成的 ID 唯一且高效至关重要。Snowflake 算法由 Twitter 开发,通过 64 位 long 型数字生成全局唯一 ID,包括 1 位标识位、41 位时间戳、10 位机器 ID 和 12 位序列号。该算法具备全局唯一性、递增性、高可用性和高性能,适用于高并发场景,如电商促销时的大量订单生成。本文介绍了使用 Go 语言的 `bwmarrin/snowflake` 和 `sony/sonyflake` 库实现 Snowflake 算法的方法。
59 1
分布式唯一ID生成:深入理解Snowflake算法在Go中的实现
|
2月前
|
算法
雪花算法反思:订单ID生成的痛点与解决方案
雪花算法(Snowflake Algorithm)因其生成唯一ID的能力而被广泛应用于分布式系统中。然而,随着业务的发展和系统规模的扩大,一些隐藏的问题逐渐浮现。本文将探讨使用雪花算法生成订单ID后可能遇到的挑战,并提供相应的解决方案。
73 2
|
2月前
|
JSON 算法 数据挖掘
基于图论算法有向图PageRank与无向图Louvain算法构建指令的方式方法 用于支撑qwen agent中的统计相关组件
利用图序列进行数据解读,主要包括节点序列分析、边序列分析以及结合节点和边序列的综合分析。节点序列分析涉及节点度分析(如入度、出度、度中心性)、节点属性分析(如品牌、价格等属性的分布与聚类)、节点标签分析(如不同标签的分布及标签间的关联)。边序列分析则关注边的权重分析(如关联强度)、边的类型分析(如管理、协作等关系)及路径分析(如最短路径计算)。结合节点和边序列的分析,如子图挖掘和图的动态分析,可以帮助深入理解图的结构和功能。例如,通过子图挖掘可以发现具有特定结构的子图,而图的动态分析则能揭示图随时间的变化趋势。这些分析方法结合使用,能够从多个角度全面解读图谱数据,为决策提供有力支持。
107 0
|
3月前
|
算法 索引
HashMap扩容时的rehash方法中(e.hash & oldCap) == 0算法推导
HashMap在扩容时,会创建一个新数组,并将旧数组中的数据迁移过去。通过(e.hash & oldCap)是否等于0,数据被巧妙地分为两类:一类保持原有索引位置,另一类索引位置增加旧数组长度。此过程确保了数据均匀分布,提高了查询效率。
51 2
|
3月前
|
搜索推荐 Shell
解析排序算法:十大排序方法的工作原理与性能比较
解析排序算法:十大排序方法的工作原理与性能比较
89 9
|
3月前
|
存储 算法 Java
数据结构与算法学习八:前缀(波兰)表达式、中缀表达式、后缀(逆波兰)表达式的学习,中缀转后缀的两个方法,逆波兰计算器的实现
前缀(波兰)表达式、中缀表达式和后缀(逆波兰)表达式的基本概念、计算机求值方法,以及如何将中缀表达式转换为后缀表达式,并提供了相应的Java代码实现和测试结果。
153 0
数据结构与算法学习八:前缀(波兰)表达式、中缀表达式、后缀(逆波兰)表达式的学习,中缀转后缀的两个方法,逆波兰计算器的实现
|
3月前
|
机器学习/深度学习 人工智能 开发框架
【AI系统】AI 学习方法与算法现状
在人工智能的历史长河中,我们见证了从规则驱动系统到现代机器学习模型的转变。AI的学习方法基于深度神经网络,通过前向传播、反向传播和梯度更新不断优化权重,实现从训练到推理的过程。当前,AI算法如CNN、RNN、GNN和GAN等在各自领域取得突破,推动技术进步的同时也带来了更大的挑战,要求算法工程师与系统设计师紧密合作,共同拓展AI技术的边界。
139 1

热门文章

最新文章