分布式系统唯一ID生成方案汇总

ゞ 浴缸里的玫瑰 2023-10-07 19:28 132阅读 0赞

4ec69a9868cc10e28052689378dbc4c8.png

点击上方蓝字关注我们

26143c922b177b01582a8cb5f6410424.png

24f2172067400df6c68e3399c4dd032a.png

系统唯一ID是我们在设计一个系统的时候常常会遇见的问题,也常常为这个问题而纠结。生成ID的方法有很多,适应不同的场景、需求以及性能要求。所以有些比较复杂的系统会有多个ID生成的策略。下面就介绍一些常见的ID生成策略。

9208e74a77d9e03c1ea6423d60cce100.png

a43569196aba500df7adbd5c47076ef2.png

1

数据库自增长序列或字段

27e02db0bb1391806e05b8a49440eeb7.png

547c9b37a390a40bd7ea9f325b409313.png

最常见的方式。利用数据库,全数据库唯一。

优点:

(1)简单,代码方便,性能可以接受。

(2)数字ID天然排序,对分页或者需要排序的结果很有帮助。

缺点:

(1)不同数据库语法和实现不同,数据库迁移的时候或多数据库版本支持的时候需要处理。

(2)在单个数据库或读写分离或一主多从的情况下,只有一个主库可以生成。有单点故障的风险。

(3)在性能达不到要求的情况下,比较难于扩展。(不适用于海量高并发)

(4)如果遇见多个系统需要合并或者涉及到数据迁移会相当痛苦。

(5)分表分库的时候会有麻烦。

(6)并非一定连续,类似MySQL,当生成新ID的事务回滚,那么后续的事务也不会再用这个ID了。这个在性能和连续性的折中。如果为了保证连续,必须要在事务结束后才能生成ID,那性能就会出现问题。

(7)在分布式数据库中,如果采用了自增主键的话,有可能会带来尾部热点。分布式数据库常常使用range的分区方式,在大量新增记录的时候,IO会集中在一个分区上,造成热点数据。

优化方案:

针对主库单点,如果有多个Master库,则每个Master库设置的起始数字不一样,步长一样,可以是Master的个数。比如:Master1 生成的是 1,4,7,10,Master2生成的是2,5,8,11 Master3生成的是 3,6,9,12。这样就可以有效生成集群中的唯一ID,也可以大大降低ID生成数据库操作的负载。

d2eefa59a19cafc2d2d0951e84e2cafe.png

4e68457cea763c149abf167435657815.png

115be93faabd7ee72a857e784029db43.png

2

UUID

c01a78f87644a174aa8b584114cacec3.png

51ff577cfe11de786ef159b3c6770b13.png

常见的方式。可以利用数据库也可以利用程序生成,一般来说全球唯一。UUID是由32个的16进制数字组成,所以每个UUID的长度是128位(16^32 = 2^128)。UUID作为一种广泛使用标准,有多个实现版本,影响它的因素包括时间、网卡MAC地址、自定义Namesapce等等。

优点:

(1)简单,代码方便。

(2)生成ID性能非常好,基本不会有性能问题。

(3)全球唯一,在遇见数据迁移,系统数据合并,或者数据库变更等情况下,可以从容应对。

缺点:

(1)没有排序,无法保证趋势递增。

(2)UUID往往是使用字符串存储,查询的效率比较低。

(3)存储空间比较大,如果是海量数据库,就需要考虑存储量的问题。

(4)传输数据量大。

(5)不可读。

edd9cbd51de71469e7475e1da6c45642.png

67745649f7923c7e726c6472479e83da.png

469e43f84d5a1b70bd870c051a81e4d2.png

3

UUID的变种

8668aa924074e3f4e6749681e0b2ee73.png

cf99b097d161b3fcbd5215d05740a6ea.png

(1)为了解决UUID不可读,可以使用UUID to Int64的方法。

f9bb28699862093116cb0fe6f85fed0f.png

(2)为了解决UUID无序的问题,NHibernate在其主键生成方式中提供了Comb算法(combined guid/timestamp)。保留GUID的10个字节,用另6个字节表示GUID生成的时间(DateTime)。

f01ebe884c05856bc86856075eea9ac6.png

用上面的算法测试一下,得到如下的结果:作为比较,前面3个是使用COMB算法得出的结果,最后12个字符串是时间序(统一毫秒生成的3个UUID),过段时间如果再次生成,则12个字符串会比图示的要大。后面3个是直接生成的GUID。

25b87e9115f8ac81f10d72369469a6e7.png如果想把时间序放在前面,可以生成后改变12个字符串的位置,也可以修改算法类的最后两个Array.Copy。

d2b3865560e9ffe7d423bafb06bc275c.png

a4ba1c6f283c8a29b3a6730be7ce98c8.png

6fb5cef43752991101ec1850a961a4a7.png

4

Redis生成ID

a499098c2efefcbc41d65c132a91a827.png

1096bf63af6bf497f96f99414a868f39.png

当使用数据库来生成ID性能不够要求的时候,我们可以尝试使用Redis来生成ID。这主要依赖于Redis是单线程的,所以也可以用生成全局唯一的ID。可以用Redis的原子操作 INCR和INCRBY来实现。可以使用Redis集群来获取更高的吞吐量。假如一个集群中有5台Redis。可以初始化每台Redis的值分别是1,2,3,4,5,然后步长都是5。各个Redis生成的ID为:

这个,随便负载到哪个机确定好,未来很难做修改。但是3-5台服务器基本能够满足器上,都可以获得不同的ID。但是步长和初始值一定需要事先需要了。使用Redis集群也可以方式单点故障的问题。

另外,比较适合使用Redis来生成每天从0开始的流水号。比如订单号=日期+当日自增长号。可以每天在Redis中生成一个Key,使用INCR进行累加。

f25e42cffa5f3619dd4606b7c0d9c1f2.png

优点:

(1)不依赖于数据库,灵活方便,且性能优于数据库。

(2)数字ID天然排序,对分页或者需要排序的结果很有帮助。

缺点:

(1)如果系统中没有Redis,还需要引入新的组件,增加系统复杂度。

(2)需要编码和配置的工作量比较大。

920b3db575127c75fcc084d7071e9341.png

0c54e46737c2834777b842c057ab09c6.png

333ecc613aa227b218c12f2f8ff1ab9d.png

5

Twitter的snowflake算法

94f078d837d38d3df3cc3c1b83eff7bb.png

b4e3dcc4968c6c57edcac4b45904e503.png

snowflake是Twitter开源的分布式ID生成算法,结果是一个long型的ID。其核心思想是:使用41bit作为毫秒数,10bit作为机器的ID(5个bit是数据中心,5个bit的机器ID),12bit作为毫秒内的流水号(意味着每个节点在每毫秒可以产生 4096 个 ID),最后还有一个符号位,永远是0。

具体实现的代码可以参看https://github.com/twitter/snowflake。雪花算法支持的TPS可以达到419万左右(2^22\*1000)。

雪花算法在工程实现上有单机版本和分布式版本。单机版本如下,分布式版本可以参看美团leaf算法:https://github.com/Meituan-Dianping/Leaf

snowflake算法可以根据自身项目的需要进行一定的修改。比如估算未来的数据中心个数,每个数据中心的机器数以及统一毫秒可以能的并发数来调整在算法中所需要的bit数。

优点:

(1)不依赖于数据库,灵活方便,且性能优于数据库。

(2)ID按照时间在单机上是递增的。

缺点:

(1)在单机上是递增的,但是由于涉及到分布式环境,每台机器上的时钟不可能完全同步,在算法上要解决时间回拨的问题。

76a0ef76947e5b4a213ae605eab77e6d.png

9af7a2b14da46e1edc7d86f22047132e.png

60d17308e588125c1aa93fe78488ebd5.png

6

利用zookeeper生成唯一ID

5892caadde9204e4c282416037c8b0a0.png

e23f64cd4d6399e04b8d3d8d59e0d105.png

zookeeper主要通过其znode数据版本来生成序列号,可以生成32位和64位的数据版本号,客户端可以使用这个版本号来作为唯一的序列号。很少会使用zookeeper来生成唯一ID。主要是由于需要依赖zookeeper,并且是多步调用API,如果在竞争较大的情况下,需要考虑使用分布式锁。因此,性能在高并发的分布式环境下,也不甚理想。

64f80b93ee4fcadc4e56bf56870fe2a5.png

ed52953da6b3287faaddc35885a097ec.png

5022810d4ab4ef339e9e95acf456e93a.png

7

MongoDB的ObjectId

69f88bba1ccebde762cd8ec0cdc5baae.png

51e2c6b84b1ea4fa4024ae08c9dddc6d.png

MongoDB的ObjectId和snowflake算法类似。它设计成轻量型的,不同的机器都能用全局唯一的同种方法方便地生成它。MongoDB 从一开始就设计用来作为分布式数据库,处理多个节点是一个核心要求。使其在分片环境中要容易生成得多。其格式如下:前4 个字节是从标准纪元开始的时间戳,单位为秒。时间戳,与随后的5 个字节组合起来,提供了秒级别的唯一性。由于时间戳在前,这意味着ObjectId 大致会按照插入的顺序排列。这对于某些方面很有用,如将其作为索引提高效率。这4 个字节也隐含了文档创建的时间。绝大多数客户端类库都会公开一个方法从ObjectId 获取这个信息。接下来的3 字节是所在主机的唯一标识符。通常是机器主机名的散列值。这样就可以确保不同主机生成不同的ObjectId,不产生冲突。为了确保在同一台机器上并发的多个进程产生的ObjectId 是唯一的,接下来的两字节来自产生ObjectId 的进程标识符(PID)。前9 字节保证了同一秒钟不同机器不同进程产生的ObjectId 是唯一的。后3 字节就是一个自动增加的计数器,确保相同进程同一秒产生的ObjectId 也是不一样的。同一秒钟最多允许每个进程拥有2563(16 777 216)个不同的ObjectId。

2f984d745be8915116d519018db414c7.png

011069ad787a6cb23228e943968a3273.png

cae47a3263d4e94ed4c7e008b7c7c6eb.png

8

TiDB的主键

7ee4ba2a75e56aed1d4f348f97c906ef.png

a4e884e9f9fbd38f0031fdf0145b5fd7.png

TiDB默认是支持自增主键的,对未声明主键的表,会提供了一个隐式主键_tidb_rowid,因为这个主键大体上是单调递增的,所以也会出现我们前面说的“尾部热点”问题。TiDB也提供了UUID函数,而且在4.0版本中还提供了另一种解决方案AutoRandom。TiDB 模仿MySQL的 AutoIncrement,提供了AutoRandom关键字用于生成一个随机ID填充指定列。

9e7fc02e221ed2d9ae6c140ab891f122.png

ae96d8dade8697c74cc1c08cbce5c38a.png

8d89ee0f3fd847a31ffc0d53ef29a438.png

原文链接:

https://www.cnblogs.com/haoxinyue/p/5208136.html

2269e488a7a2f38c067fc7d924db9d75.png

点个在看你最好看

3cc9257327c66a30dc5ab42067e7f61f.png

发表评论

表情:
评论列表 (有 0 条评论,132人围观)

还没有评论,来说两句吧...

相关阅读

    相关 分布式唯一ID生成方案

    ​ 在应用程序中,经常需要全局唯一的`ID`作为数据库主键。如何生成全局唯一ID? ​ 首先,需要确定全局唯一`ID`是整型还是字符串?如果是字符串,那么现有的`UUID`就