Go基于雪花算法生成隨機(jī)id
雪花算法
雪花算法是twitter開(kāi)源的由64位整數(shù)組成的分布式ID,性能高,并且在單機(jī)上遞增。
1.第一位占用1bit,其值始終是0,沒(méi)有實(shí)際作用。
2.時(shí)間戳占用41bit,單位為毫秒,總共可以容納約69年的時(shí)間。當(dāng)然,我們的時(shí)間毫秒計(jì)數(shù)不會(huì)真的從1970年開(kāi)始記,那樣我們的系統(tǒng)跑到2039/9/7 23:47:35 就不能用了,所以這里的時(shí)間戳只是相對(duì)于某個(gè)時(shí)間的增量,比如我們的系統(tǒng)上線是2023-03-17,那么我們完全可以把這個(gè)timestamp當(dāng)作是從2023-03-17 00:00:00. 000的偏移量。
3.工作機(jī)器id占用10bit,其中高位5bit是數(shù)據(jù)中心ID,低位5bit是工作節(jié)點(diǎn)ID,最多可以容納1024個(gè)節(jié)點(diǎn)。
4.序列號(hào)占用12bit,用來(lái)記錄同毫秒內(nèi)產(chǎn)生的不同id。每個(gè)節(jié)點(diǎn)每毫秒0開(kāi)始不斷累加,最多可以累加到4095,同一毫秒-共可以產(chǎn)生4096個(gè)ID。SnowFlake算法在同一毫秒內(nèi)做多可以產(chǎn)生多少全局唯一ID呢?同一毫秒的ID數(shù)量 = 1024 X 4096 = 4194304
Go實(shí)現(xiàn)代碼
1.github.com/bwmarrin/snowflake
package main import ( "fmt" "github.com/bwmarrin/snowflake" "time" ) var node *snowflake.Node //初始化一個(gè)node func Init(startTime string, machineID int64) (err error) { //自定義開(kāi)始時(shí)間 var st time.Time st, err = time.Parse("2006-01-02", startTime) if err != nil { return } snowflake.Epoch = st.UnixNano() / 1000000 node, err = snowflake.NewNode(machineID) return } func GenID() int64 { return node.Generate().Int64() } func main() { if err := Init("2023-03-17", 1); err != nil { fmt.Printf("init failed, err:%v\n", err) return } id := GenID() fmt.Println(id) } var ( sonyFlake *sonyflake.Sonyflake // 實(shí)例 sonyMachineID uint16 //機(jī)器ID ) func getMachineID() (uint16, error) { //返回全局定義的機(jī)器ID return sonyMachineID, nil }
2.github.com/sony/sonyflake
package main import ( "fmt" "github.com/sony/sonyflake" "time" ) var ( sonyFlake *sonyflake.Sonyflake // 實(shí)例 sonyMachineID uint16 //機(jī)器ID ) func getMachineID() (uint16, error) { //返回全局定義的機(jī)器ID return sonyMachineID, nil } //需傳入當(dāng)前的機(jī)器ID func Init(machineId uint16) (err error) { sonyMachineID = machineId t, _ := time.Parse(" 2006-01-02", "2023-03-17") //初始化一個(gè)開(kāi)始的時(shí)間 settings := sonyflake.Settings{ // 生成全局配置 StartTime: t, MachineID: getMachineID, //指定機(jī)器ID } sonyFlake = sonyflake.NewSonyflake(settings) // 用配置生成sonyflake節(jié)點(diǎn) return } // GetID返回生成的id值 func GetID() (id uint64, err error) { //拿到sonyflake節(jié) 點(diǎn)生成id值 if sonyFlake == nil { err = fmt.Errorf("snoy flake not inited") return } id, err = sonyFlake.NextID() return }
到此這篇關(guān)于Go基于雪花算法生成隨機(jī)id的文章就介紹到這了,更多相關(guān)Go 生成隨機(jī)id內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
用go gin server來(lái)做文件上傳服務(wù)
今天小編就為大家分享一篇關(guān)于用go gin server來(lái)做文件上傳服務(wù),小編覺(jué)得內(nèi)容挺不錯(cuò)的,現(xiàn)在分享給大家,具有很好的參考價(jià)值,需要的朋友一起跟隨小編來(lái)看看吧2019-04-04服務(wù)器端Go程序?qū)﹂L(zhǎng)短鏈接的處理及運(yùn)行參數(shù)的保存
這篇文章主要介紹了服務(wù)器端Go程序?qū)﹂L(zhǎng)短鏈接的處理及運(yùn)行參數(shù)的保存,這里針對(duì)使用Go語(yǔ)言編寫(xiě)的Socket服務(wù)器進(jìn)行實(shí)例說(shuō)明,需要的朋友可以參考下2016-03-03Go語(yǔ)言實(shí)現(xiàn)Viper配置管理筆記
Viper 是一個(gè)功能強(qiáng)大、靈活易用的配置管理工具,本文主要介紹了Go語(yǔ)言實(shí)現(xiàn)Viper配置管理筆記,具有一定的參考價(jià)值,感興趣的可以了解一下2025-04-04初學(xué)Go必備的vscode插件及最常用快捷鍵和代碼自動(dòng)補(bǔ)全
這篇文章主要給大家介紹了關(guān)于初學(xué)vscode寫(xiě)Go必備的vscode插件及最常用快捷鍵和代碼自動(dòng)補(bǔ)全的相關(guān)資料,由于vscode是開(kāi)源免費(fèi)的,而且開(kāi)發(fā)支持vscode的插件相對(duì)比較容易,更新速度也很快,需要的朋友可以參考下2023-07-07Go語(yǔ)言使用GORM操作數(shù)據(jù)庫(kù)使用指南
GORM(全稱為Go?Object?Relational?Mapping)是一個(gè)在Go語(yǔ)言中使用的輕量級(jí)的對(duì)象關(guān)系映射(ORM)庫(kù),本文主要為大家介紹了GORM操作數(shù)據(jù)庫(kù)具體方法,需要的可以參考一下2023-05-05Go語(yǔ)言基礎(chǔ)語(yǔ)法之結(jié)構(gòu)體及方法詳解
結(jié)構(gòu)體類型可以用來(lái)保存不同類型的數(shù)據(jù),也可以通過(guò)方法的形式來(lái)聲明它的行為。本文將介紹go語(yǔ)言中的結(jié)構(gòu)體和方法,以及“繼承”的實(shí)現(xiàn)方法2021-09-09Go中時(shí)間與時(shí)區(qū)問(wèn)題的深入講解
go語(yǔ)言中如果不設(shè)置指定的時(shí)區(qū),通過(guò)time.Now()獲取到的就是本地時(shí)區(qū),下面這篇文章主要給大家介紹了關(guān)于Go中時(shí)間與時(shí)區(qū)問(wèn)題的相關(guān)資料,需要的朋友可以參考下2021-12-12一文帶你感受Go語(yǔ)言空結(jié)構(gòu)體的魔力
在?Go?語(yǔ)言中,有一種特殊的用法可能讓許多人感到困惑,那就是空結(jié)構(gòu)體,本文將對(duì)Go空結(jié)構(gòu)體進(jìn)行詳解,準(zhǔn)備一杯你最喜歡的飲料或茶,隨著本文一探究竟吧2023-05-05