网易首页 > 网易号 > 正文 申请入驻

分布式应用中唯一主键生成解析

0
分享至

分布式服务中经常会遇到这样的业务场景:

l一些服务发送消息到队列,另一些服务从队列消费消息,消息可能会重复,消费端需要做幂等,为了达到业务的幂等,希望有一个不能重复且趋势递增的编号存在;

l分库分表中需要生成不能重复且趋势递增的产品编号或者订单编号;

为此,生成编号的机制需要满足如下条件:

l局部、全局唯一。

l趋势递增。

工作中有很多方式可以生成唯一且趋势递增的id编号, 适应不同的场景、需求以及性能要求。所以有些比较复杂的系统会有多个ID生成的策略。下面就介绍一些常见的ID生成策略。

一、数据库自增长序列或字段

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

优点:

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

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

缺点:

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

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

3)在性能达不到要求的情况下,比较难于扩展。

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

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

优化方案:

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

封装成单个服务,一次生成一定数量的编号存储到数据库表或者缓存中,需要id编号的服务调用生成服务挨个获取。

二、UUID

常见的方式。可以利用数据库也可以利用程序生成,一般来说全球唯一。

优点:

1)简单,代码方便。

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

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

缺点:

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

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

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

4)传输数据量大

5)不可读。

三、 UUID的变种

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

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

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

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

四. Redis生成ID

当使用数据库来生成ID性能不够要求的时候,我们可以尝试使用Redis来生成ID。这主要依赖于Redis是单线程的,所以也可以用生成全局唯一的ID。可以用Redis的原子操作 INCR和INCRBY来实现。

可以使用Redis集群来获取更高的吞吐量。假如一个集群中有5台Redis。可以初始化每台Redis的值分别是1,2,3,4,5,然后步长都是5。各个Redis生成的ID为:

A:1,6,11,16,21

B:2,7,12,17,22

C:3,8,13,18,23

D:4,9,14,19,24

E:5,10,15,20,25

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

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

优点:

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

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

缺点:

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

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

五. Twitter的snowflake算法

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

测试代码如下:

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

优点:

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

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

缺点:

1)在单机上是递增的,但是由于涉及到分布式环境,每台机器上的时钟不可能完全同步,也许有时候也会出现不是全局递增的情况。

六. 利用zookeeper生成唯一ID

zookeeper主要通过其znode数据版本来生成序列号,可以生成32位和64位的数据版本号,客户端可以使用这个版本号来作为唯一的序列号。

很少会使用zookeeper来生成唯一ID。主要是由于需要依赖zookeeper,并且是多步调用API,如果在竞争较大的情况下,需要考虑使用分布式锁。因此,性能在高并发的分布式环境下,也不甚理想。

七. MongoDB的ObjectId

MongoDB的ObjectId和snowflake算法类似。它设计成轻量型的,不同的机器都能用全局唯一的同种方法方便地生成它。MongoDB 从一开始就设计用来作为分布式数据库,处理多个节点是一个核心要求。使其在分片环境中要容易生成得多。

其格式如下:

前4 个字节是从标准纪元开始的时间戳,单位为秒。时间戳,与随后的5 个字节组合起来,提供了秒级别的唯一性。由于时间戳在前,这意味着ObjectId 大致会按照插入的顺序排列。这对于某些方面很有用,如将其作为索引提高效率。这4 个字节也隐含了文档创建的时间。绝大多数客户端类库都会公开一个方法从ObjectId 获取这个信息。

接下来的3 字节是所在主机的唯一标识符。通常是机器主机名的散列值。这样就可以确保不同主机生成不同的ObjectId,不产生冲突。

为了确保在同一台机器上并发的多个进程产生的ObjectId 是唯一的,接下来的两字节来自产生ObjectId 的进程标识符(PID)。

前9 字节保证了同一秒钟不同机器不同进程产生的ObjectId 是唯一的。后3 字节就是一个自动增加的计数器,确保相同进程同一秒产生的ObjectId 也是不一样的。同一秒钟最多允许每个进程拥有2563(16 777 216)个不同的ObjectId。

特别声明:以上内容(如有图片或视频亦包括在内)为自媒体平台“网易号”用户上传并发布,本平台仅提供信息存储服务。

Notice: The content above (including the pictures and videos if any) is uploaded and posted by a user of NetEase Hao, which is a social media platform and only provides information storage services.

相关推荐
热点推荐
上海19号线设站全分析,建设时间落地,宝山、虹口、闵行赢麻了

上海19号线设站全分析,建设时间落地,宝山、虹口、闵行赢麻了

妮子说美食
2024-11-28 00:01:38
退伍军人救下跳河母女,带到家中安顿,准备离开时,军人母亲说

退伍军人救下跳河母女,带到家中安顿,准备离开时,军人母亲说

华人星光
2024-11-27 14:43:36
刚刚,中办、国办重磅印发!

刚刚,中办、国办重磅印发!

21世纪经济报道
2024-11-27 21:20:02
医生直言:喝茶少的人,肝脏要比喝茶过多的人更强健?是真的吗?

医生直言:喝茶少的人,肝脏要比喝茶过多的人更强健?是真的吗?

今日养生之道
2024-11-21 14:27:06
权贵家庭还愿意托举女婿吗?看了网友的分享,不寒而栗

权贵家庭还愿意托举女婿吗?看了网友的分享,不寒而栗

有趣的火烈鸟
2024-11-26 06:51:26
“铁人”王进喜后来怎么样了,结局很令人意外

“铁人”王进喜后来怎么样了,结局很令人意外

顾史
2024-11-19 14:22:10
韩网热议:张元英稍微长一点肉,就无敌了!

韩网热议:张元英稍微长一点肉,就无敌了!

文娱没有圈
2024-11-11 11:15:53
妹子更喜欢18cm还是40分钟?女司机带经历回答太赤裸裸了,看完秒懂哈哈

妹子更喜欢18cm还是40分钟?女司机带经历回答太赤裸裸了,看完秒懂哈哈

经典段子
2024-08-19 22:46:36
被普京政府说中了:美日绝密计划曝光,五角大楼已为中国选好战场

被普京政府说中了:美日绝密计划曝光,五角大楼已为中国选好战场

千里持剑
2024-11-25 13:47:25
袈裟下的资本有多惊人?身披16万袈裟,名下18家企业,存款超30亿

袈裟下的资本有多惊人?身披16万袈裟,名下18家企业,存款超30亿

橘子大娱社
2024-11-25 21:30:03
中国十大特级战斗英雄,排名不分先后,谁才是你心目中最崇拜的人

中国十大特级战斗英雄,排名不分先后,谁才是你心目中最崇拜的人

芳芳历史烩
2024-11-17 13:12:01
40岁保姆色诱70岁教授,“性猝死”背后真相是如何露馅的?

40岁保姆色诱70岁教授,“性猝死”背后真相是如何露馅的?

苏大强专栏
2024-07-26 11:50:51
央行等七部门:鼓励金融机构充分运用内外部数据和大数据技术

央行等七部门:鼓励金融机构充分运用内外部数据和大数据技术

证券时报
2024-11-27 18:28:11
100岁叶嘉莹追悼会大咖如云,女儿从加拿大回国,已四世同堂!

100岁叶嘉莹追悼会大咖如云,女儿从加拿大回国,已四世同堂!

古希腊掌管松饼的神
2024-11-27 20:56:55
她靠这张脸统一全球审美,被歧视了20年的笨蛋美人终于翻身了!

她靠这张脸统一全球审美,被歧视了20年的笨蛋美人终于翻身了!

新氧
2024-11-26 11:20:48
冬天使劲吃,一降脂,二利尿,三养颜,买50斤晒成干,特省事!

冬天使劲吃,一降脂,二利尿,三养颜,买50斤晒成干,特省事!

阿龙美食记
2024-11-27 21:14:36
看了“兔子警官”的升职真相,才懂反诈老陈的后悔,破防的不是钱

看了“兔子警官”的升职真相,才懂反诈老陈的后悔,破防的不是钱

阿矗论古今
2024-11-25 16:03:52
随着火箭险胜森林狼 湖人惨败太阳 NBA西部排名如下

随着火箭险胜森林狼 湖人惨败太阳 NBA西部排名如下

阿柒体讯
2024-11-28 02:19:15
乌军空袭库拉霍沃俄军弹药库!俄军用滑板车发起冲锋?

乌军空袭库拉霍沃俄军弹药库!俄军用滑板车发起冲锋?

项鹏飞
2024-11-26 20:53:14
看吧!这就是国内的某企业,有没有惊到你?

看吧!这就是国内的某企业,有没有惊到你?

慧翔百科
2024-11-11 11:42:35
2024-11-28 03:19:00
非常美丽的句子12138
非常美丽的句子12138
每天分享有趣的视频
6063文章数 10253关注度
往期回顾 全部

科技要闻

利润暴跌40%后,三星芯片部门任命新负责人

头条要闻

前实习生攻击公司模型训练 字节跳动起诉索赔800余万

头条要闻

前实习生攻击公司模型训练 字节跳动起诉索赔800余万

体育要闻

网红主播踢顶级联赛 这是什么奇葩俱乐部

娱乐要闻

天呐鹿晗瘦了这么多,像漫画出来的

财经要闻

马光远:所有的房地产政策基本都到位了

汽车要闻

中国的超豪华车尊界S800 预售百万起/ L3级智驾

态度原创

健康
教育
手机
时尚
军事航空

花18万治疗阿尔茨海默病,值不值?

教育要闻

孩子需要怎样的鼓励?专家解析,这种鼓励方式更加贴心有效

手机要闻

2499元起!REDMI K80发布:骁龙8至尊版,性价比超强

10种秋冬配色方案,有效摆脱路人感!

军事要闻

黎以停火协议生效 以军警告

无障碍浏览 进入关怀版