日本综合一区二区|亚洲中文天堂综合|日韩欧美自拍一区|男女精品天堂一区|欧美自拍第6页亚洲成人精品一区|亚洲黄色天堂一区二区成人|超碰91偷拍第一页|日韩av夜夜嗨中文字幕|久久蜜综合视频官网|精美人妻一区二区三区

RELATEED CONSULTING
相關(guān)咨詢
選擇下列產(chǎn)品馬上在線溝通
服務(wù)時(shí)間:8:30-17:00
你可能遇到了下面的問題
關(guān)閉右側(cè)工具欄

新聞中心

這里有您想知道的互聯(lián)網(wǎng)營(yíng)銷解決方案
帶你吃透幾種大廠分布式ID設(shè)計(jì)方案

帶你吃透幾種大廠分布式id設(shè)計(jì)方案

作者:無聊 2021-06-04 20:09:19
網(wǎng)絡(luò)
通信技術(shù)
分布式 最近公司在擴(kuò)招后端高級(jí)開發(fā),有幸成為面試官之一,其中問的最多一個(gè)問題就是分布式ID的幾種解決方案,不客氣的說前身小公司的開發(fā)答得完整的很少。

創(chuàng)新互聯(lián)公司堅(jiān)持“要么做到,要么別承諾”的工作理念,服務(wù)領(lǐng)域包括:成都網(wǎng)站建設(shè)、成都做網(wǎng)站、企業(yè)官網(wǎng)、英文網(wǎng)站、手機(jī)端網(wǎng)站、網(wǎng)站推廣等服務(wù),滿足客戶于互聯(lián)網(wǎng)時(shí)代的元江縣網(wǎng)站設(shè)計(jì)、移動(dòng)媒體設(shè)計(jì)的需求,幫助企業(yè)找到有效的互聯(lián)網(wǎng)解決方案。努力成為您成熟可靠的網(wǎng)絡(luò)建設(shè)合作伙伴!

本文轉(zhuǎn)載自微信公眾號(hào)「無聊學(xué)Java」,作者無聊。轉(zhuǎn)載本文請(qǐng)聯(lián)系無聊學(xué)Java公眾號(hào)。

前言

最近公司在擴(kuò)招后端高級(jí)開發(fā),有幸成為面試官之一,其中問的最多一個(gè)問題就是分布式ID的幾種解決方案,不客氣的說前身小公司的開發(fā)答得完整的很少。

于是就抽出了周末的時(shí)間整理了幾種主流的分布式ID生成方案,希望能夠幫助到你們。

開篇幾個(gè)問題

1. 為什么需要分布式全局唯一ID以及分布式ID的業(yè)務(wù)需求

在復(fù)雜分布式系統(tǒng)中,往往需要對(duì)大量的數(shù)據(jù)和消息進(jìn)行唯一標(biāo)識(shí)。

  • 如在美團(tuán)點(diǎn)評(píng)的金融、支付、餐飲、酒店等業(yè)務(wù)場(chǎng)景
  • 貓眼電影等產(chǎn)品的系統(tǒng)中數(shù)據(jù)日漸增長(zhǎng),對(duì)數(shù)據(jù)分庫分表后需要有一個(gè)唯一ID來表示一條數(shù)據(jù)或者消息。
  • 特別一點(diǎn)的如訂單、騎手、優(yōu)惠劵也都需要一個(gè)唯一ID做為標(biāo)識(shí)。

此時(shí)一個(gè)能生成唯一ID的系統(tǒng)是非常必要的。

2. ID生成規(guī)則部分硬性要求

  • 全局唯一:既然是唯一標(biāo)識(shí),那么全局唯一是最基本的要求。
  • 趨勢(shì)遞增:在MySQL的InnoDB引擎中使用的是聚集索引,由于多數(shù)RDBMS使用Btree的數(shù)據(jù)結(jié)構(gòu)來存儲(chǔ)索引數(shù)據(jù),在主鍵的選擇上面我們應(yīng)該盡量使用有序的主鍵來保證寫入性能。
  • 單調(diào)遞增:保證下一個(gè)ID一定大于上一個(gè)ID,例如事務(wù)版本號(hào)、IM增量消息、排序等特殊需求。
  • 信息安全:如果ID是連續(xù)的,那么惡意用戶的扒取工作就非常容易做了,直接按照順序下載指定URL即可;如果是訂單號(hào)那么更加危險(xiǎn),競(jìng)爭(zhēng)對(duì)手可以知道我們一天的單量;所以在一些應(yīng)用場(chǎng)景下,需要ID無規(guī)則不規(guī)則,讓競(jìng)爭(zhēng)對(duì)手不好猜。
  • 含時(shí)間戳:這樣就能在開發(fā)中快速了解這個(gè)分布式ID的生成時(shí)間。

3. ID生成系統(tǒng)的可用性要求

  • 高可用:發(fā)一個(gè)獲取分布式ID的請(qǐng)求,服務(wù)器就要保證99.999%的情況下給我創(chuàng)建一個(gè)唯一分布式ID
  • 低延遲:發(fā)一個(gè)獲取分布式ID的請(qǐng)求,服務(wù)器就要快,極速
  • 高QPS:假如并發(fā)一口氣10萬個(gè)創(chuàng)建分布式ID請(qǐng)求同時(shí)過來,服務(wù)器需要頂?shù)米∏页晒?chuàng)建10萬個(gè)分布式ID

通用的幾種方案

隨著系統(tǒng)架構(gòu)以及業(yè)務(wù)的演變,分布式ID生成也是有N中解決方案,以下就簡(jiǎn)單的列舉幾種。

1. UUID

這種方案估計(jì)大家都了解,最簡(jiǎn)單的一種方案。

  
 
 
 
  1. public static void main(String[] args) { 
  2.     String uuid = UUID.randomUUID().toString(); 
  3.     System.out.println(uuid); 

如果只是考慮唯一性,那么UUID基本可以滿足需求。

缺點(diǎn)

  • 無序:無法預(yù)測(cè)他的生成順序,不能生成遞增有序的數(shù)字
  • 主鍵:ID作為主鍵時(shí)在特定的環(huán)境下會(huì)存在一些問題,比如做DB主鍵的場(chǎng)景下,UUID非常不適用,MySQL官方有明確的建議主鍵要盡量越短越好,36位的UUID不合要求。
  • 索引:會(huì)導(dǎo)致B+樹索引的分裂。

2. 數(shù)據(jù)庫自增主鍵

此種方案有一定的局限性,在高并發(fā)集群上此策略不可用。

3. 基于Redis生成全局ID策略

  • 因?yàn)镽edis是單線程,天生保證原子性,所以可以使用INCR和INCRBY來實(shí)現(xiàn)。
  • 集群分布式

在Redis集群下,同樣和MySQL一樣需要設(shè)置不同的增長(zhǎng)步數(shù),同時(shí)key需要設(shè)置有效期;可以使用Redis集群來獲取更高的吞吐量;假如一個(gè)集群中有五個(gè)Redis,那么初始化每臺(tái)Redis步長(zhǎng)分別是1,2,3,4,5,然后步長(zhǎng)都是5。

4. snowflake(雪花算法)

  • 推特的雪花算法生成ID能夠按照時(shí)間有序生成。
  • 雪花算法生成ID的結(jié)果是一個(gè)64bit大小的整數(shù),為一個(gè)Long型(轉(zhuǎn)換為字符串后長(zhǎng)度最多19)
  • 分布式系統(tǒng)內(nèi)不會(huì)產(chǎn)生ID碰撞(由datecenter和workerId作區(qū)分),并且效率較高。

結(jié)構(gòu)

雪花算法的幾個(gè)核心組成部分如下圖:

號(hào)段解析

  • 1bit符號(hào)位:不用,因?yàn)槎M(jìn)制最高位是符號(hào)位,1表示負(fù)數(shù),0表示正數(shù),生成的id一般都是用正數(shù),所以最高位固定位0
  • 41bit時(shí)間戳,用于記錄時(shí)間戳,毫秒級(jí)
    • 41位可以表示2^41 - 1個(gè)數(shù)字
    • 如果只用來表示正整數(shù)(計(jì)算機(jī)正數(shù)包含0),可以表示的數(shù)值范圍是0-2^41 - 1,減一是因?yàn)榭杀硎镜臄?shù)值范圍是從0開始算的,而不是1
    • 也就是說41位可以表示2^41 - 1個(gè)毫秒的值,轉(zhuǎn)換為單位年則是69年。
  • 10bit工作進(jìn)程位,用于記錄工作機(jī)器id
    • 可以部署在2^10 = 1024個(gè)節(jié)點(diǎn),包括五位datacenterId和五位workerId
    • 五位可以表示的最大整數(shù)位2^5 - 1 = 31,即可以使用0,1,2…31這32個(gè)數(shù)字來表示不同的datacenterId和workerId
  • 12bit序列號(hào),序列號(hào),用來記錄同毫秒內(nèi) 產(chǎn)生的不同的ID
    • 12bit可以表示的最大正整數(shù)位2^12 - 1 = 4095,即可以表示0,1….4094這4095個(gè)數(shù)字
    • 表示同一機(jī)器同一時(shí)間戳(毫秒)中產(chǎn)生的4095個(gè)ID序號(hào)

優(yōu)點(diǎn)

  • 所有生成的id按時(shí)間趨勢(shì)遞增
  • 整個(gè)分布式內(nèi)不會(huì)產(chǎn)生重復(fù)id,因?yàn)橛衐atacenterId和workerId來做區(qū)分。
  • 毫秒數(shù)在高位,自增序列在低位,整個(gè)ID都是趨勢(shì)遞增的
  • 不依賴數(shù)據(jù)庫、redis等第三方系統(tǒng),以服務(wù)的方式部署,穩(wěn)定性更高,生成ID的性能也是非常高的。
  • 可以根據(jù)自身業(yè)務(wù)分配bit位,非常靈活。

缺點(diǎn)

  • 依賴機(jī)器時(shí)鐘,如果機(jī)器時(shí)鐘回退,會(huì)導(dǎo)致重復(fù)ID生成
  • 在單機(jī)上是遞增的,但是由于設(shè)計(jì)到分布式環(huán)境,每臺(tái)機(jī)器上的時(shí)鐘不可能完全同步,有時(shí)候會(huì)出現(xiàn)不是全局遞增的情況。(此缺點(diǎn)可以認(rèn)為蕪鎖胃,一般分布式ID只要求趨勢(shì)遞增,并不會(huì)嚴(yán)格要求遞增,90%的需求都只需要趨勢(shì)遞增)

源碼

  
 
 
 
  1. /** 
  2.  * twitter的snowflake算法 -- java實(shí)現(xiàn) 
  3.  *  
  4.  * @author beyond 
  5.  * @date 2016/11/26 
  6.  */ 
  7. public class SnowFlake { 
  8.  
  9.     /** 
  10.      * 起始的時(shí)間戳 
  11.      */ 
  12.     private final static long START_STMP = 1480166465631L; 
  13.  
  14.     /** 
  15.      * 每一部分占用的位數(shù) 
  16.      */ 
  17.     private final static long SEQUENCE_BIT = 12; //序列號(hào)占用的位數(shù) 
  18.     private final static long MACHINE_BIT = 5;   //機(jī)器標(biāo)識(shí)占用的位數(shù) 
  19.     private final static long DATACENTER_BIT = 5;//數(shù)據(jù)中心占用的位數(shù) 
  20.  
  21.     /** 
  22.      * 每一部分的最大值 
  23.      */ 
  24.     private final static long MAX_DATACENTER_NUM = -1L ^ (-1L << DATACENTER_BIT); 
  25.     private final static long MAX_MACHINE_NUM = -1L ^ (-1L << MACHINE_BIT); 
  26.     private final static long MAX_SEQUENCE = -1L ^ (-1L << SEQUENCE_BIT); 
  27.  
  28.     /** 
  29.      * 每一部分向左的位移 
  30.      */ 
  31.     private final static long MACHINE_LEFT = SEQUENCE_BIT; 
  32.     private final static long DATACENTER_LEFT = SEQUENCE_BIT + MACHINE_BIT; 
  33.     private final static long TIMESTMP_LEFT = DATACENTER_LEFT + DATACENTER_BIT; 
  34.  
  35.     private long datacenterId;  //數(shù)據(jù)中心 
  36.     private long machineId;     //機(jī)器標(biāo)識(shí) 
  37.     private long sequence = 0L; //序列號(hào) 
  38.     private long lastStmp = -1L;//上一次時(shí)間戳 
  39.  
  40.     public SnowFlake(long datacenterId, long machineId) { 
  41.         if (datacenterId > MAX_DATACENTER_NUM || datacenterId < 0) { 
  42.             throw new IllegalArgumentException("datacenterId can't be greater than MAX_DATACENTER_NUM or less than 0"); 
  43.         } 
  44.         if (machineId > MAX_MACHINE_NUM || machineId < 0) { 
  45.             throw new IllegalArgumentException("machineId can't be greater than MAX_MACHINE_NUM or less than 0"); 
  46.         } 
  47.         this.datacenterId = datacenterId; 
  48.         this.machineId = machineId; 
  49.     } 
  50.  
  51.     /** 
  52.      * 產(chǎn)生下一個(gè)ID 
  53.      * 
  54.      * @return 
  55.      */ 
  56.     public synchronized long nextId() { 
  57.         long currStmp = getNewstmp(); 
  58.         if (currStmp < lastStmp) { 
  59.             throw new RuntimeException("Clock moved backwards.  Refusing to generate id"); 
  60.         } 
  61.  
  62.         if (currStmp == lastStmp) { 
  63.             //相同毫秒內(nèi),序列號(hào)自增 
  64.             sequence = (sequence + 1) & MAX_SEQUENCE; 
  65.             //同一毫秒的序列數(shù)已經(jīng)達(dá)到最大 
  66.             if (sequence == 0L) { 
  67.                 currStmp = getNextMill(); 
  68.             } 
  69.         } else { 
  70.             //不同毫秒內(nèi),序列號(hào)置為0 
  71.             sequence = 0L; 
  72.         } 
  73.  
  74.         lastStmp = currStmp; 
  75.  
  76.         return (currStmp - START_STMP) << TIMESTMP_LEFT //時(shí)間戳部分 
  77.                 | datacenterId << DATACENTER_LEFT       //數(shù)據(jù)中心部分 
  78.                 | machineId << MACHINE_LEFT             //機(jī)器標(biāo)識(shí)部分 
  79.                 | sequence;                             //序列號(hào)部分 
  80.     } 
  81.  
  82.     private long getNextMill() { 
  83.         long mill = getNewstmp(); 
  84.         while (mill <= lastStmp) { 
  85.             mill = getNewstmp(); 
  86.         } 
  87.         return mill; 
  88.     } 
  89.  
  90.     private long getNewstmp() { 
  91.         return System.currentTimeMillis(); 
  92.     } 
  93.  
  94.     public static void main(String[] args) { 
  95.         SnowFlake snowFlake = new SnowFlake(2, 3); 
  96.  
  97.         for (int i = 0; i < (1 << 12); i++) { 
  98.             System.out.println(snowFlake.nextId()); 
  99.         } 
  100.  
  101.     } 

測(cè)試

  
 
 
 
  1. //測(cè)試使用雪花算法生成ID 
  2. //構(gòu)造函數(shù)中傳入datacenterId和workerId 
  3. SnowFlake snowFlake = new SnowFlake(1,1); 
  4. for (int i = 0; i < 10; i++) { 
  5.     long id = snowFlake.nextId(); 
  6.     System.out.println("id:" + id + "\t" + String.valueOf(id).length() + "位"); 
  7.     System.out.println("------------------------------------------"); 

Spring Boot整合雪花算法

引入hutool-all,maven依賴引入如下:

  
 
 
 
  1.  
  2.      
  3.         cn.hutool 
  4.         hutool-all 
  5.         5.4.2 
  6.      
  7.      
  8.         org.springframework.boot 
  9.         spring-boot-starter-web 
  10.         2.2.1.RELEASE 
  11.      
  12.      
  13.         org.projectlombok 
  14.         lombok 
  15.         1.18.16 
  16.      
  17.  

創(chuàng)建一個(gè)SnowFlake配置類

  
 
 
 
  1. @Configuration 
  2. public class SnowFlakeConfig { 
  3.     @Value("${application.datacenterId}") 
  4.     private Long datacenterId; 
  5.     @Value("${application.workerId}") 
  6.     private Long workerId; 
  7.  
  8.     /*** 
  9.      * 注入一個(gè)生成雪花ID的對(duì)象 
  10.      * @return 
  11.      */ 
  12.     @Bean 
  13.     public Snowflake snowflake() { 
  14.         return new Snowflake(workerId,datacenterId); 
  15.     } 

yml配置文件:

  
 
 
 
  1. application: 
  2.   datacenterId: 2 
  3.   workerId: 1 
  4. server: 
  5.   port: 7777 

service 層:

  
 
 
 
  1. @Service 
  2. public class OrderService { 
  3.     @Autowired 
  4.     private Snowflake snowflake; 
  5.  
  6.     public String getIdBySnowFlake() { 
  7.         return String.valueOf(snowflake.nextId()); 
  8.     } 

其他開源的解決方案

很多大廠都對(duì)雪花算法做出了改進(jìn),開源了一些改進(jìn)方案,如下:

  • 百度開源的分布式唯一ID生成器UidGenerator
  • Leaf–美團(tuán)點(diǎn)評(píng)分布式ID生成系統(tǒng)

當(dāng)前文章:帶你吃透幾種大廠分布式ID設(shè)計(jì)方案
文章路徑:http://www.dlmjj.cn/article/copicii.html