溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊(cè)×
其他方式登錄
點(diǎn)擊 登錄注冊(cè) 即表示同意《億速云用戶服務(wù)條款》

怎樣通過(guò)雪花算法用Python實(shí)現(xiàn)一個(gè)簡(jiǎn)單的發(fā)號(hào)器

發(fā)布時(shí)間:2021-02-05 11:05:21 來(lái)源:億速云 閱讀:375 作者:小新 欄目:開(kāi)發(fā)技術(shù)

這篇文章將為大家詳細(xì)講解有關(guān)怎樣通過(guò)雪花算法用Python實(shí)現(xiàn)一個(gè)簡(jiǎn)單的發(fā)號(hào)器,小編覺(jué)得挺實(shí)用的,因此分享給大家做個(gè)參考,希望大家閱讀完這篇文章后可以有所收獲。

實(shí)現(xiàn)一個(gè)簡(jiǎn)單的發(fā)號(hào)器

根據(jù)snowflake算法的原理實(shí)現(xiàn)一個(gè)簡(jiǎn)單的發(fā)號(hào)器,產(chǎn)生不重復(fù)、自增的id。

1.snowflake算法的簡(jiǎn)單描述

怎樣通過(guò)雪花算法用Python實(shí)現(xiàn)一個(gè)簡(jiǎn)單的發(fā)號(hào)器

這里的snowflake算法是用二進(jìn)制的,有64位。其中41位的時(shí)間戳表示:當(dāng)前時(shí)間戳減去某個(gè)設(shè)定的起始時(shí)間,10位標(biāo)識(shí)表示:不同的機(jī)器、數(shù)據(jù)庫(kù)的標(biāo)識(shí)ID等等,序列號(hào)為每秒或每毫秒內(nèi)自增的id。

我做的時(shí)候沒(méi)有用位運(yùn)算去實(shí)現(xiàn),而是做了一個(gè)十進(jìn)制的,16位的(當(dāng)時(shí)項(xiàng)目要求是16位的)。但是實(shí)現(xiàn)發(fā)號(hào)器的基本策略是一樣的,通過(guò)時(shí)間戳和標(biāo)識(shí)來(lái)防止重復(fù),通過(guò)序列號(hào)實(shí)現(xiàn)自增。當(dāng)然啦,重點(diǎn)不是發(fā)號(hào)器多少位,而是根據(jù)項(xiàng)目的實(shí)際情況,利用snowflake算法的原理,實(shí)現(xiàn)一個(gè)適合自己項(xiàng)目的發(fā)號(hào)器。

2.Python實(shí)現(xiàn)

時(shí)間戳:9位,起始時(shí)間為2018-01-01 00:00:00 ,時(shí)間戳為當(dāng)前時(shí)間減去起始時(shí)間。時(shí)間戳有9為,可用時(shí)間為 999999999/(606024*365)≈31(年)。

標(biāo)識(shí)ID:2位,我用的時(shí)候比較簡(jiǎn)單,只是涉及一個(gè)數(shù)據(jù)庫(kù)的情況,所以用一張數(shù)據(jù)表對(duì)應(yīng)一個(gè)標(biāo)識(shí)ID,可用100張表。

序列號(hào):5位,我時(shí)間戳用的是秒級(jí)的,但是5位是10萬(wàn)個(gè)序列號(hào),經(jīng)過(guò)測(cè)試在一秒內(nèi)是完全夠用的。

所以時(shí)間戳、標(biāo)識(shí)ID、序列號(hào)的位數(shù)也沒(méi)規(guī)定說(shuō)一定要多少,根據(jù)自己項(xiàng)目的實(shí)際來(lái)即可。

代碼如下:

import time
class MySnow:
  def __init__(self,dataID):
    self.start = int(time.mktime(time.strptime('2018-01-01 00:00:00', "%Y-%m-%d %H:%M:%S")))
    self.last = int(time.time())
    self.countID = 0
    self.dataID = dataID  # 數(shù)據(jù)ID,這個(gè)自定義或是映射

  def get_id(self):
    # 時(shí)間差部分
    now = int(time.time())
    temp = now-self.start
    if len(str(temp)) < 9: # 時(shí)間差不夠9位的在前面補(bǔ)0
      length = len(str(temp))
      s = "0" * (9-length)
      temp = s + str(temp)
    if now == self.last:
      self.countID += 1  # 同一時(shí)間差,序列號(hào)自增
    else:
      self.countID = 0  # 不同時(shí)間差,序列號(hào)重新置為0
      self.last = now
    # 標(biāo)識(shí)ID部分
    if len(str(self.dataID)) < 2:
      length = len(str(self.dataID))
      s = "0" * (2-length)
      self.dataID = s + str(self.dataID)
    # 自增序列號(hào)部分
    if self.countID == 99999: # 序列號(hào)自增5位滿了,睡眠一秒鐘
      time.sleep(1)
    countIDdata = str(self.countID)
    if len(countIDdata) < 5: # 序列號(hào)不夠5位的在前面補(bǔ)0
      length = len(countIDdata)
      s = "0"*(5-length)
      countIDdata = s + countIDdata
    id = str(temp) + str(self.dataID) + countIDdata
    return id

使用方法:

snow = MySnow(dataID="00")
print(snow.get_id())

其中dataID即為標(biāo)識(shí)ID,countID為自增序列號(hào)。dataID可以一個(gè)通過(guò)自定義的映射表獲得,這個(gè)視實(shí)際的項(xiàng)目情況而定。

3.關(guān)于并發(fā)

首先,直接用這個(gè)發(fā)號(hào)器是不能進(jìn)行并發(fā)操作,會(huì)產(chǎn)生重復(fù)的id。如果真的要進(jìn)行并發(fā),那么就要權(quán)衡一下并發(fā)和位數(shù)的哪個(gè)更重要了!

拿實(shí)際例子來(lái)說(shuō)吧,比如我并發(fā)的目的是為了節(jié)省時(shí)間,讓程序更快的跑完,這時(shí)候?yàn)榱瞬l(fā),我把dataID中拿出一位來(lái),標(biāo)識(shí)不同的子進(jìn)程,這樣可以防止產(chǎn)生重復(fù)的id。但是實(shí)際上這用了位數(shù)去換取時(shí)間,如果是id位數(shù)比較少的情況,比如16位的,dataID比較少,我個(gè)人認(rèn)為這樣是不值得的,有些奢侈。這時(shí)候便是位數(shù)比并發(fā)重要啦。

當(dāng)時(shí)如果位數(shù)充裕,比如20位的,需要并發(fā)就并發(fā)啦。

還有一種實(shí)現(xiàn)并發(fā)的方法,就是給發(fā)號(hào)器加鎖,發(fā)號(hào)的時(shí)候加鎖,發(fā)完了解鎖。這個(gè)我沒(méi)有試過(guò),有興趣的可以試一下哈哈。但是我有個(gè)疑惑啊,就是不斷加鎖和解鎖切換,帶來(lái)的時(shí)間和資源開(kāi)銷(xiāo)會(huì)不會(huì)很大呢。

關(guān)于“怎樣通過(guò)雪花算法用Python實(shí)現(xiàn)一個(gè)簡(jiǎn)單的發(fā)號(hào)器”這篇文章就分享到這里了,希望以上內(nèi)容可以對(duì)大家有一定的幫助,使各位可以學(xué)到更多知識(shí),如果覺(jué)得文章不錯(cuò),請(qǐng)把它分享出去讓更多的人看到。

向AI問(wèn)一下細(xì)節(jié)

免責(zé)聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點(diǎn)不代表本網(wǎng)站立場(chǎng),如果涉及侵權(quán)請(qǐng)聯(lián)系站長(zhǎng)郵箱:is@yisu.com進(jìn)行舉報(bào),并提供相關(guān)證據(jù),一經(jīng)查實(shí),將立刻刪除涉嫌侵權(quán)內(nèi)容。

AI