Loading

分布式唯一键生成浅析

今天跟大家简单介绍下分布式唯一键浅析

背景:

  随着微服务概念的兴起,我们系统拆分的越来越模块化,系统变多了,调用链路页变长了,那么问题就出来了,一次用户的请求过来,我如何能够在分布式的系统中跟踪这次调用的路径 或者一次调用如何能够生成一个全局的唯一ID

举例说明一下:

  以我们系统为例:我们现在唯一键生成有两种方式(1)、使用sequenceUtil生成(2)、生成UUID

  首先说使用sequenceUtil生成的,目前我们订单的唯一键的生成方式是这样,首先要保证存在表sequence_value,这个表里只有两个字段,id、name;每次请求到sequenceUtil 生成唯一主键,都会在本机中存一个map,该map中key是表名,value中包含 唯一键开始值、唯一键跨度值、唯一键当前值,最大值

当我们应用调用sequence得到唯一键时,若是应用中没有这个key,需要先往表中插入一条唯一键数据,id值为0,然后更新为200 ,然后这个机器能够使用的主键的范围就是从0到200,当200个主键用完之后再继续累加。

好处:使用预展的方式,我们每次生成id都知道该id具体值的内容

坏处:由于每个机器之间有200的跨度,出现问题的几率页不大, 但是可能会出现两个机器都使用完,同时发起请求,由于在sql中存在CAS校验,有可能无法变更id的值,导致主键id生成失败

  第二种方式:String uuid = UUID.randomUUID().toString(), 这种方式可以生成一个长度为32位的全局唯一识别码,目前我们的结算页请求地址中的requestId就是这种方式生成的,以及我们再tp_order中的accessGuid也是这种方式生成的。

我们可以看到他可以保证全局唯一,而且代码简单,性能也很好,但是不足在于长度较长,而且它在作为数据库索引存储时,由于数据库的索引大都是B+树的方式,当无序的数插入时,可能会导致因为把新记录插入到合适的位置而移动大量的数据,从而降低了写入性能。

  第三种方式:snowFlake算法,也是十分著名的分布式ID生成的算法,是一种划分命名空间来生成ID的一种算法,结算是一个long型的ID。其核心是把64-bit分别划分为多段。 这个算法现在不维护, 但是git项目还在,大家可以下载玩一下

                                      (图片来自:https://crazyfzw.github.io/2018/07/21/unique-id-generate/)

  • 1位标识符:始终是0,由于long基本类型在Java中是带符号的,最高位是符号位,正数是0,负数是1,所以id一般是正数,最高位是0。
  • 41位时间戳:41位时间截不是存储当前时间的时间截,而是存储时间截的差值(当前时间截 - 开始时间截 )得到的值,这里的的开始时间截,一般是我们的id生成器开始使用的时间,由我们程序来指定的。
  • 10位机器标识码:可以部署在1024个节点,如果机器分机房(IDC)部署,这10位可以由 5位机房ID + 5位机器ID 组成。
  • 12位序列:毫秒内的计数,12位的计数顺序号支持每个节点每毫秒(同一机器,同一时间截)产生4096个ID序号 

优点:

  • 时间戳在高位,自增序列在低位,整个ID是趋势递增的,按照时间有序递增。
  • 可以根据bit位不同划分不同业务

缺点:

  • 依赖机器的时钟,如果服务器时钟回拨,会导致重复ID生成。
  • 在分布式环境上,每个服务器的时钟不可能完全同步,有时会出现不是全局递增的情况。

 分布式生成唯一键就说到这里,我们再来稍微提一嘴各个公司的分布式追踪系统,分布式追踪系统是基于分布式唯一键生成的基础上,对系统中的调用链路进行跟踪。

京东的:Hydra - 京东开源的基于Dubbo的调用分布跟踪系统

    CallGraph 、pfinder

美团的

Leaf-segment 

 https://tech.meituan.com/2017/04/21/mt-leaf.html

 

Flickr 在分布式系统中我们可以多部署几台机器,每台机器设置不同的初始值,且步长和机器数相等。比如有两台机器。设置步长step为2,TicketServer1的初始值为1(1,3,5,7,9,11…)、TicketServer2的初始值为2(2,4,6,8,10…)。这是Flickr团队在2010年撰文介绍的一种主键生成策略

 

微信:https://blog.csdn.net/hellojackjiang2011/article/details/82997656

 

大家有兴趣可以继续研究这方面的东西。加油!

posted @ 2019-06-12 15:14  冯廷鑫  阅读(682)  评论(0编辑  收藏  举报