欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

Java分布式ID中Snowflake雪花算法應(yīng)用實現(xiàn)

 更新時間:2024年07月30日 11:32:53   作者:dazhong2012  
Snowflake算法作為一種高效且易于實現(xiàn)的分布式ID生成方案,能夠很好地滿足分布式系統(tǒng)中對全局唯一ID的需求,本文就來介紹一下Java分布式ID中Snowflake雪花算法應(yīng)用實現(xiàn),感興趣的可以了解一下

在構(gòu)建分布式系統(tǒng)時,如何生成全局唯一的ID是一個重要且常見的挑戰(zhàn)。ID需要具有全局唯一性、遞增性以及趨勢遞增性,以便在分布式環(huán)境中進行有效的數(shù)據(jù)存儲、查詢和分片。本文將詳細解析Java分布式ID的生成方案,并深入探討Snowflake算法的原理及其在Spring Boot中的應(yīng)用。

一、分布式ID生成方案概述

在分布式系統(tǒng)中,常見的ID生成方案包括UUID、數(shù)據(jù)庫自增ID、Redis生成ID以及Snowflake算法等。其中,Snowflake算法以其高效、簡潔的特性受到廣泛關(guān)注。Snowflake算法生成的ID是一個64位的整數(shù),由時間戳、工作機器ID和序列號組成,保證了全局唯一性、遞增性和趨勢遞增性。

二、Snowflake算法詳解

Snowflake算法的核心思想是將一個64位的整數(shù)劃分為多個部分,用于記錄不同的信息。具體來說,Snowflake算法將64位ID分為以下幾部分:

在這里插入圖片描述

  • 未使用位:最高位是符號位,由于生成的ID都是正數(shù),所以最高位固定為0。
  • 時間戳差值:占據(jù)41位,記錄當(dāng)前時間與開始時間的差值(單位:毫秒)。這部分可以支持系統(tǒng)運行69年。
  • 工作機器ID:包括數(shù)據(jù)中心ID和工作節(jié)點ID,各占5位,總共10位。這樣,Snowflake算法最多可以支持1024個節(jié)點。
  • 序列號:占12位,用于記錄同一毫秒內(nèi)產(chǎn)生的不同ID。這保證了在同一機器、同一時間戳下,可以生成最多4096個不同的ID。

三、Snowflake算法在Spring Boot中的應(yīng)用

在Spring Boot項目中集成Snowflake算法生成唯一ID,通常我們會創(chuàng)建一個ID生成器服務(wù),并在需要的地方注入使用。以下是一個簡單的示例:

import org.springframework.stereotype.Service;  
  
@Service  
public class SnowflakeIdWorker {  
    // 初始化參數(shù)  
    private final long twepoch = 1288834974657L;  
    private long workerId;  
    private long datacenterId;  
    private long sequence = 0L;  
  
    private final long workerIdBits = 5L;  
    private final long datacenterIdBits = 5L;  
    private final long maxWorkerId = -1L ^ (-1L << workerIdBits);  
    private final long maxDatacenterId = -1L ^ (-1L << datacenterIdBits);  
    private final long sequenceBits = 12L;  
  
    private final long workerIdShift = sequenceBits;  
    private final long datacenterIdShift = sequenceBits + workerIdBits;  
    private final long timestampLeftShift = sequenceBits + workerIdBits + datacenterIdBits;  
    private final long sequenceMask = -1L ^ (-1L << sequenceBits);  
  
    private long lastTimestamp = -1L;  
  
    public SnowflakeIdWorker(long workerId, long datacenterId) {  
        if (workerId > maxWorkerId || workerId < 0) {  
            throw new IllegalArgumentException(String.format("worker Id can't be greater than %d or less than 0", maxWorkerId));  
        }  
        if (datacenterId > maxDatacenterId || datacenterId < 0) {  
            throw new IllegalArgumentException(String.format("datacenter Id can't be greater than %d or less than 0", maxDatacenterId));  
        }  
        this.workerId = workerId;  
        this.datacenterId = datacenterId;  
    }  
  
    public synchronized long nextId() {  
        long timestamp = timeGen();  
  
        if (timestamp < lastTimestamp) {  
            throw new RuntimeException(String.format("Clock moved backwards.  Refusing to generate id for %d milliseconds", lastTimestamp - timestamp));  
        }  
  
        if (lastTimestamp == timestamp) {  
            sequence = (sequence + 1) & sequenceMask;  
            if (sequence == 0) {  
                timestamp = tilNextMillis(lastTimestamp);  
            }  
        } else {  
            sequence = 0L;  
        }  
  
        lastTimestamp = timestamp;  
  
        return ((timestamp - twepoch) << timestampLeftShift) |  
                (datacenterId << datacenterIdShift) |  
                (workerId << workerIdShift) |  
                sequence;  
    }  
  
    protected long tilNextMillis(long lastTimestamp) {  
        long timestamp = timeGen();  
        while (timestamp <= lastTimestamp) {  
            timestamp = timeGen();  
        }
   		return timestamp;
	}

	protected long timeGen() {  
	    return System.currentTimeMillis();  
	}
}

四、Spring Boot中使用Snowflake

在Spring Boot中,你可以創(chuàng)建一個配置類來初始化SnowflakeIdWorker,并在需要生成ID的地方注入使用。

import org.springframework.context.annotation.Bean;  
import org.springframework.context.annotation.Configuration;  
  
@Configuration  
public class SnowflakeConfig {  
  
    @Bean  
    public SnowflakeIdWorker snowflakeIdWorker() {  
        // 根據(jù)實際部署情況設(shè)置工作機器ID和數(shù)據(jù)中心ID  
        long workerId = 1L;  
        long datacenterId = 1L;  
        return new SnowflakeIdWorker(workerId, datacenterId);  
    }  
}

在需要使用ID的地方,你可以通過依賴注入的方式獲取SnowflakeIdWorker實例,并調(diào)用nextId()方法生成唯一ID。

import org.springframework.beans.factory.annotation.Autowired;  
import org.springframework.stereotype.Service;  
  
@Service  
public class SomeService {  
  
    private final SnowflakeIdWorker snowflakeIdWorker;  
  
    @Autowired  
    public SomeService(SnowflakeIdWorker snowflakeIdWorker) {  
        this.snowflakeIdWorker = snowflakeIdWorker;  
    }  
  
    public void someMethod() {  
        long id = snowflakeIdWorker.nextId();  
        // 使用生成的ID進行后續(xù)操作  
    }  
}

五、總結(jié)

在Spring中,Snowflake算法作為一種高效且易于實現(xiàn)的分布式ID生成方案,能夠很好地滿足分布式系統(tǒng)中對全局唯一ID的需求。通過合理配置工作機器ID和數(shù)據(jù)中心ID,并結(jié)合Spring的依賴注入特性,可以方便地在Spring Boot項目中集成Snowflake算法,為業(yè)務(wù)邏輯提供穩(wěn)定可靠的ID生成服務(wù)。需要注意的是,在使用Snowflake算法時,應(yīng)確保工作機器ID和數(shù)據(jù)中心ID的唯一性,以避免ID沖突。此外,由于Snowflake算法依賴于系統(tǒng)時鐘,因此在時鐘回撥的情況下可能會導(dǎo)致ID生成異常,需要妥善處理這種情況。

到此這篇關(guān)于Java分布式ID中Snowflake雪花算法應(yīng)用實現(xiàn)的文章就介紹到這了,更多相關(guān)Java Snowflake雪花算法內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!

相關(guān)文章

  • 如何處理器攔截器(HandlerInterceptor)

    如何處理器攔截器(HandlerInterceptor)

    這篇文章主要介紹了如何處理器攔截器(HandlerInterceptor)問題,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教
    2023-07-07
  • mybatis-plus主鍵策略生成失敗的解決

    mybatis-plus主鍵策略生成失敗的解決

    本文主要介紹了mybatis-plus主鍵策略生成失敗的解決,文中通過示例代碼介紹的非常詳細,對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧
    2023-03-03
  • java String的深入理解

    java String的深入理解

    這篇文章主要介紹了java String的深入理解的相關(guān)資料,希望通過本文大家能理解String的用法,需要的朋友可以參考下
    2017-09-09
  • 如何使用GSON解析JSON數(shù)據(jù)

    如何使用GSON解析JSON數(shù)據(jù)

    這篇文章主要介紹了如何使用GSON解析JSON數(shù)據(jù),具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教
    2022-08-08
  • SpringBoot詳細講解異步任務(wù)如何獲取HttpServletRequest

    SpringBoot詳細講解異步任務(wù)如何獲取HttpServletRequest

    在使用框架日常開發(fā)中需要在controller中進行一些異步操作減少請求時間,但是發(fā)現(xiàn)在使用@Anysc注解后會出現(xiàn)Request對象無法獲取的情況,本文就此情況給出完整的解決方案
    2022-04-04
  • IDEA中application.properties的圖標(biāo)顯示不正常的問題及解決方法

    IDEA中application.properties的圖標(biāo)顯示不正常的問題及解決方法

    這篇文章主要介紹了IDEA中application.properties的圖標(biāo)顯示不正常的問題及解決方法,本文給大家介紹的非常詳細,對大家的學(xué)習(xí)或工作具有一定的參考借鑒價值,需要的朋友可以參考下
    2021-04-04
  • Java并發(fā)編程之線程狀態(tài)介紹

    Java并發(fā)編程之線程狀態(tài)介紹

    這篇文章主要介紹了Java并發(fā)編程之線程狀態(tài),當(dāng)線程被創(chuàng)建并啟動以后,它既不是一啟動就進入了執(zhí)行狀態(tài),也不是一直處于執(zhí)行狀態(tài),下面和小編一起進入文章了解具體的相關(guān)介紹吧
    2022-04-04
  • SpringBoot響應(yīng)處理之以Json數(shù)據(jù)返回的實現(xiàn)方法

    SpringBoot響應(yīng)處理之以Json數(shù)據(jù)返回的實現(xiàn)方法

    這篇文章主要介紹了SpringBoot整合Web開發(fā)其中Json數(shù)據(jù)返回的實現(xiàn),文中通過示例代碼介紹的非常詳細,對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧
    2022-09-09
  • SpringBoot使用AOP與注解實現(xiàn)請求參數(shù)自動填充流程詳解

    SpringBoot使用AOP與注解實現(xiàn)請求參數(shù)自動填充流程詳解

    面向切面編程(aspect-oriented programming,AOP)主要實現(xiàn)的目的是針對業(yè)務(wù)處理過程中的切面進行提取,諸如日志、事務(wù)管理和安全這樣的系統(tǒng)服務(wù),從而使得業(yè)務(wù)邏輯各部分之間的耦合度降低,提高程序的可重用性,同時提高了開發(fā)的效率
    2023-02-02
  • Java Socket實現(xiàn)猜數(shù)字小游戲

    Java Socket實現(xiàn)猜數(shù)字小游戲

    這篇文章主要為大家詳細介紹了Java Socket實現(xiàn)猜數(shù)字小游戲,文中示例代碼介紹的非常詳細,具有一定的參考價值,感興趣的小伙伴們可以參考一下
    2020-09-09

最新評論