分布式系統(tǒng)的唯一id生成算法

背景:如果分庫分表場景下,id不唯一准谚,就不能通過主鍵索引來查詢了详囤。

1桨武、獨立數(shù)據(jù)庫自增id

這個方案就是說你的系統(tǒng)每次要生成一個 id肋拔,都是往一個獨立庫的一個獨立表里插入一條沒什 么業(yè)務含義的數(shù)據(jù),然后獲取一個數(shù)據(jù)庫自增的一個 id呀酸。拿到這個 id 之后再往對應的分庫分表 里去寫入。 假如說你有一個 auto_id 庫里有一個叫做 auto_id的 表琼梆,有一個 id 是自增的性誉。 那么你每次要獲取一個全局唯一 id,直接往這個表里插入一條記錄茎杂,獲取一個全局唯一 id 即 可错览,然后這個全局唯一 id 就可以插入訂單的分庫分表中。 這個方案的好處就是方便簡單煌往,誰都會用倾哺。缺點就是單庫生成自增 id,要是搞并發(fā)的話刽脖,就會 有瓶頸的羞海,因為 auto_id 庫要是承載個每秒幾萬并發(fā),肯定是不現(xiàn)實的了曲管。

2却邓、UUID

用UUID 生成一個全局唯一的 id。
好處就是每個系統(tǒng)本地生成院水,不要基于數(shù)據(jù)庫來了
不好之處就是腊徙,uuid 太長了(32位),作為主鍵性能太差了檬某,不適合用于主鍵撬腾。 如果你是要隨機生成個什么文件名了,編號之類的恢恼,你可以用 uuid民傻,但是作為主鍵是不能用 uuid 的。

3厅瞎、獲取系統(tǒng)當前時間

這個方案的意思就是獲取當前時間作為全局唯一的 id饰潜。 但是問題是,并發(fā)很高的時候和簸,比如1秒并發(fā)幾千彭雾,會有重復的情況,這個是肯定不合適的锁保。 一般如果使用這個方案薯酝,是將當前時間跟很多其他的業(yè)務字段拼接起來半沽,作為一個 id,如果業(yè)務 上可以接受吴菠,那么也是可以的者填。 可以將別的業(yè)務字段值跟當前時間拼接起來,組成一個全局唯一的編號做葵,比如說訂單編號: 時間戳 + 用戶 id + 業(yè)務含義編碼占哟;或者 時間戳+6個隨機數(shù)(26個英文字母+數(shù)字)
同時,表索引可以添加唯一索引酿矢。

4榨乎、snowflake 算法

snowflake 算法,是 twitter 開源的分布式 id 生成算法
其核心思想就是:使用一個 64 bit 的 long 型的數(shù)字作為全局唯一 id瘫筐,這 64 個 bit 中蜜暑,其中 1 個 bit 是不用的,然后用其中的 41 bit 作為毫秒數(shù)策肝,用10 bit 作為工作機器 id肛捍,12 bit 作為序列 號。

image.png

第一部分之众,是 1 個 bit:0拙毫,這個是無意義的(因為二進制里第一個 bit 為如果是 1,那么都是負數(shù)酝枢,但是我們生成的 id 都是正數(shù)恬偷,所 以第一個 bit 統(tǒng)一都是 0)
第二個部分是 41 個 bit:表示的是時間戳,單位是毫秒(41 bit 可以表示的數(shù)字多達 2^41 - 1帘睦,也就是可以標識 2 ^ 41 - 1 個毫秒值袍患,換算成年就 是表示 69 年的時間。)
第三個部分是 5 個 bit:表示的是機房 id竣付,10001 (最多代表2 ^ 5個機房(32 個機房))
第四個部分是 5 個 bit:表示的是機器 id诡延,1 1001 (每個機房?可以代表2 ^ 5個機器(32臺機器))
第五個部分是 12 個 bit:表示的序號,就是某個機房某臺機器上這一毫秒內(nèi)同時生成的 id 的序號古胆,0000 00000000(12 bit可以代表的最大正整數(shù)是2 ^ 12 - 1 = 4096肆良,也就是說可以用這個12bit代表的數(shù)字 來區(qū)分同1個毫秒內(nèi)的4096個不同的id)

簡單來說,你的某個服務假設要生成1個全局唯一id逸绎,那么就可以發(fā)送1個請求給部署了 snowflake算法的系統(tǒng)惹恃,由這個snowflake算法系統(tǒng)來生成唯一id。

snowflake算法系統(tǒng)接收到這個請求之后棺牧,首先就會用二進制位運算的方式生成1個64 bit 的long型id巫糙,64個bit中的第1個bit是無意義的。 接著41個bit颊乘,就可以用當前時間戳(單位到毫秒)参淹,然后接著5個bit設置上這個機房id醉锄,還有5 個bit設置上機器id。 最后再判斷一下浙值,當前這臺機房的這臺機器上這1毫秒內(nèi)恳不,這是第幾個請求,給這次生成id的 請求累加一個序號开呐,作為最后的12個bit烟勋。 最終一個64個bit的id就出來了,類似于上圖中那樣负蚊。
這個算法可以保證說神妹,一個機房的一臺機器上,在同一毫秒內(nèi)家妆,生成了一個唯一的 id∶崦可能一 個毫秒內(nèi)會生成多個 id伤极,但是有最后 12 個 bit 的序號來區(qū)分開來∫躺耍總之就是用一個 64bit 的數(shù)字中各個 bit 位來設置不同的標志位哨坪,區(qū)分每一個 id。

image.png

算法實現(xiàn)如下

public class IdWorker { 
private long workerId; // 這個就是代表了機器id 
private long datacenterId; // 這個就是代表了機房id 
private long sequence; // 這個就是代表了1毫秒內(nèi)生成的多個id的最新序號 
public IdWorker(long workerId, long datacenterId, long sequence) { 

// 要求就是你傳遞進來的機房id和機器id不能超過32乍楚,不能小于于0 
if (workerId > maxWorkerId || workerId < 0) { 
  throw new IllegalArgumentException( String.format("worker Id can't be greater than %d or less t "))
}

if (datacenterId > maxDatacenterId || datacenterId < 0) { 
throw new IllegalArgumentException( String.format("datacenter Id can't be greater than %d or le "))
}

this.workerId = workerId; 
this.datacenterId = datacenterId; 
this.sequence = sequence; 
}

private long twepoch = 1288834974657L; 
private long workerIdBits = 5L; 
private long datacenterIdBits = 5L;

// 這個是二進制運算当编,就是5 bit最多只能有31個數(shù)字,也就是說機器id最多只能是32以內(nèi) 
private long maxWorkerId = -1L ^ (-1L << workerIdBits); 
// 這個是一個意思徒溪,就是5 bit最多只能有31個數(shù)字忿偷,機房id最多只能是32以內(nèi) 
private long maxDatacenterId = -1L ^ (-1L << datacenterIdBits); 
private long sequenceBits = 12L; 
private long workerIdShift = sequenceBits; 
private long datacenterIdShift = sequenceBits + workerIdBits; 
private long timestampLeftShift = sequenceBits + workerIdBits + datacen 
private long sequenceMask = -1L ^ (-1L << sequenceBits); 
private long lastTimestamp = -1L; 
public long getWorkerId(){ 
return workerId; 
}
public long getDatacenterId() { 
return datacenterId; 
}
public long getTimestamp() { 
return System.currentTimeMillis(); 
}

// 這個是核心算法,通過調(diào)調(diào)nextId()方法臊泌,讓當前這臺機器上的snowflake算法程序生成1一個id
 public synchronized long nextId() { 
// 獲取當前時間戳鲤桥,單位是毫秒
 long timestamp = timeGen();
 if (timestamp < lastTimestamp) {
 System.err.printf( "clock is moving backwards. Rejecting requests until %d.", throw new RuntimeException( String.format("Clock moved backwards. Refusing to generate lastTimestamp - timestamp))"));
 }

// 假設在同1個毫秒內(nèi)又發(fā)送了一個請求生成1個id 
// 這個時候就得把seqence序號給遞增1,最多就是4096 
if (lastTimestamp == timestamp) {
// 這個意思是說1個毫秒內(nèi)最多只能有4096個數(shù)字渠概,無論你傳遞多少進來茶凳, 
//這個位運算保證始終就是在4096這個范圍內(nèi),避免傳遞的sequence超過了4096
sequence = (sequence + 1) & sequenceMask;
 if (sequence == 0) {
 timestamp = tilNextMillis(lastTimestamp);
 } 
} else {
 sequence = 0;
 }
// 記錄最近1次生成id的時間戳播揪,單位是毫秒
 lastTimestamp = timestamp;
 // 最核心的二進制位運算操作贮喧,生成1個64bit的id 
// 先將當前時間戳左移,放到41 bit那猪狈;將機房id左移放到5 bit那箱沦;將機器id左移 
// 最后拼接起來成一個64 bit的二進制數(shù)字,轉(zhuǎn)換成10進制就是個long型 
return ((timestamp - twepoch) << timestampLeftShift) | (datacenterId << datacenterIdShift) | (workerId << workerIdShift) | sequence; 
}
private long tilNextMillis(long lastTimestamp) {
 long timestamp = timeGen();
 while (timestamp <= lastTimestamp) {
 timestamp = timeGen();
 }return timestamp;
 }
private long timeGen(){
 return System.currentTimeMillis();
 }
//---------------測試--------------- 
public static void main(String[] args) {
 IdWorker worker = new IdWorker(1,1,1);
 for (int i = 0; i < 30; i++) {
System.out.println(worker.nextId()); 
      }
   }
}

在實際的開發(fā)中罪裹,這個 snowflake 算法可以做小點點改進饱普。 我們在生成唯一 id 的時候运挫,肯定都需要指定一個表名,比如說訂單表 的唯一 id套耕。 所以上面那 64 個 bit 中谁帕,代表機房的那 5 個 bit,可以使用業(yè)務表名稱來替代冯袍,比如 00001 代表的是訂單表匈挖。 因為其實很多時候,機房并沒有那么多康愤,所以那 5 個 bit 做做機房 id 意義不是太大儡循。 這樣就可以做到,snowflake 算法系統(tǒng)的每1臺機器征冷,對一個業(yè)務表择膝,在某1毫秒內(nèi),可生成
1個唯1的 id检激,1毫秒內(nèi)生成很多 id肴捉,用最后 12 個 bit 來區(qū)分序號對待。

?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末叔收,一起剝皮案震驚了整個濱河市齿穗,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌饺律,老刑警劉巖窃页,帶你破解...
    沈念sama閱讀 217,277評論 6 503
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場離奇詭異复濒,居然都是意外死亡脖卖,警方通過查閱死者的電腦和手機,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 92,689評論 3 393
  • 文/潘曉璐 我一進店門芝薇,熙熙樓的掌柜王于貴愁眉苦臉地迎上來胚嘲,“玉大人,你說我怎么就攤上這事洛二〔雠” “怎么了?”我有些...
    開封第一講書人閱讀 163,624評論 0 353
  • 文/不壞的土叔 我叫張陵晾嘶,是天一觀的道長妓雾。 經(jīng)常有香客問我,道長垒迂,這世上最難降的妖魔是什么械姻? 我笑而不...
    開封第一講書人閱讀 58,356評論 1 293
  • 正文 為了忘掉前任,我火速辦了婚禮机断,結(jié)果婚禮上楷拳,老公的妹妹穿的比我還像新娘绣夺。我一直安慰自己,他們只是感情好欢揖,可當我...
    茶點故事閱讀 67,402評論 6 392
  • 文/花漫 我一把揭開白布陶耍。 她就那樣靜靜地躺著,像睡著了一般她混。 火紅的嫁衣襯著肌膚如雪烈钞。 梳的紋絲不亂的頭發(fā)上,一...
    開封第一講書人閱讀 51,292評論 1 301
  • 那天坤按,我揣著相機與錄音毯欣,去河邊找鬼。 笑死臭脓,一個胖子當著我的面吹牛酗钞,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播来累,決...
    沈念sama閱讀 40,135評論 3 418
  • 文/蒼蘭香墨 我猛地睜開眼算吩,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了佃扼?” 一聲冷哼從身側(cè)響起,我...
    開封第一講書人閱讀 38,992評論 0 275
  • 序言:老撾萬榮一對情侶失蹤蔼夜,失蹤者是張志新(化名)和其女友劉穎兼耀,沒想到半個月后,有當?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體求冷,經(jīng)...
    沈念sama閱讀 45,429評論 1 314
  • 正文 獨居荒郊野嶺守林人離奇死亡瘤运,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 37,636評論 3 334
  • 正文 我和宋清朗相戀三年,在試婚紗的時候發(fā)現(xiàn)自己被綠了匠题。 大學時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片拯坟。...
    茶點故事閱讀 39,785評論 1 348
  • 序言:一個原本活蹦亂跳的男人離奇死亡,死狀恐怖韭山,靈堂內(nèi)的尸體忽然破棺而出郁季,到底是詐尸還是另有隱情,我是刑警寧澤钱磅,帶...
    沈念sama閱讀 35,492評論 5 345
  • 正文 年R本政府宣布梦裂,位于F島的核電站,受9級特大地震影響盖淡,放射性物質(zhì)發(fā)生泄漏年柠。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點故事閱讀 41,092評論 3 328
  • 文/蒙蒙 一褪迟、第九天 我趴在偏房一處隱蔽的房頂上張望冗恨。 院中可真熱鬧答憔,春花似錦、人聲如沸掀抹。這莊子的主人今日做“春日...
    開封第一講書人閱讀 31,723評論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽渴丸。三九已至侯嘀,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間谱轨,已是汗流浹背戒幔。 一陣腳步聲響...
    開封第一講書人閱讀 32,858評論 1 269
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留土童,地道東北人诗茎。 一個月前我還...
    沈念sama閱讀 47,891評論 2 370
  • 正文 我出身青樓,卻偏偏與公主長得像献汗,于是被迫代替她去往敵國和親敢订。 傳聞我的和親對象是個殘疾皇子,可洞房花燭夜當晚...
    茶點故事閱讀 44,713評論 2 354

推薦閱讀更多精彩內(nèi)容