溫馨提示×

溫馨提示×

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

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

mybatis-plus雪花算法自動(dòng)生成機(jī)器id原理的示例分析

發(fā)布時(shí)間:2021-06-04 14:54:29 來源:億速云 閱讀:458 作者:小新 欄目:開發(fā)技術(shù)

這篇文章主要介紹了mybatis-plus雪花算法自動(dòng)生成機(jī)器id原理的示例分析,具有一定借鑒價(jià)值,感興趣的朋友可以參考下,希望大家閱讀完這篇文章之后大有收獲,下面讓小編帶著大家一起了解一下。

1、雪花算法原理

        雪花算法使用一個(gè) 64 bit 的 long 型的數(shù)字作為全局唯一 id。這 64 個(gè) bit 中,其中 1 個(gè) bit 是不用的,然后用其中的 41 bit 作為毫秒數(shù),用 10 bit 作為工作機(jī)器 id,12 bit 作為序列號。

mybatis-plus雪花算法自動(dòng)生成機(jī)器id原理的示例分析

  1. 1bit,不用,因?yàn)槎M(jìn)制中最高位是符號位,1表示負(fù)數(shù),0表示正數(shù)。生成的id一般都是用整數(shù),所以最高位固定為0。

  2. 41bit-時(shí)間戳,用來記錄時(shí)間戳,毫秒級。

  3. 10bit-工作機(jī)器id,用來記錄工作機(jī)器id。

  4. 12bit-序列號,序列號,用來記錄同毫秒內(nèi)產(chǎn)生的不同id。即可以用0、1、2、3、…4094這4095個(gè)數(shù)字,來表示同一機(jī)器同一時(shí)間截(毫秒)內(nèi)產(chǎn)生的4095個(gè)ID序號。

SnowFlake可以保證:

所有生成的id按時(shí)間趨勢遞增
整個(gè)分布式系統(tǒng)內(nèi)不會(huì)產(chǎn)生重復(fù)id(因?yàn)橛衐atacenterId和workerId來做區(qū)分)

        如上大概了解了雪花算法的原理,而且也知道機(jī)器號對于雪花算法的重要性。如果機(jī)器號一樣,可能會(huì)出現(xiàn)id重復(fù)的情況。

        mybatis-plus自3.3.0開始,默認(rèn)使用雪花算法+UUID(不含中劃線),但是它并沒有強(qiáng)制讓開發(fā)者配置機(jī)器號。這一點(diǎn)很是疑惑,這樣可能會(huì)讓不了解雪花算法的人埋下了一個(gè)坑。
        但是這么強(qiáng)大的一個(gè)框架難道真的沒有做優(yōu)化嗎?帶著問題,查看了下mybatis-plus雪花算法源碼com.baomidou.mybatisplus.core.toolkit.Sequence。最終發(fā)現(xiàn)在沒有設(shè)置機(jī)器號的情況下,會(huì)通過當(dāng)前物理網(wǎng)卡地址和jvm的進(jìn)程ID自動(dòng)生成。這真的是一個(gè)較好的解決方案。一般在一個(gè)集群中,MAC+JVM進(jìn)程PID一樣的幾率非常小。

2、自動(dòng)生成唯一機(jī)器號源碼

核心代碼。有兩個(gè)構(gòu)造方法,一個(gè)無參構(gòu)造,一個(gè)有參構(gòu)造。

public Sequence() {
    //通過當(dāng)前物理網(wǎng)卡地址獲取datacenterId
    this.datacenterId = getDatacenterId(maxDatacenterId);
    //物理網(wǎng)卡地址+jvm進(jìn)程pi獲取workerId
    this.workerId = getMaxWorkerId(datacenterId, maxWorkerId);
}

/**
 * 有參構(gòu)造器
 *
 * @param workerId     工作機(jī)器 ID
 * @param datacenterId 序列號
 */
public Sequence(long workerId, long datacenterId) {
    Assert.isFalse(workerId > maxWorkerId || workerId < 0,
            String.format("worker Id can't be greater than %d or less than 0", maxWorkerId));
    Assert.isFalse(datacenterId > maxDatacenterId || datacenterId < 0,
            String.format("datacenter Id can't be greater than %d or less than 0", maxDatacenterId));
    this.workerId = workerId;
    this.datacenterId = datacenterId;
}
  • 無參構(gòu)造 開發(fā)者沒有設(shè)置機(jī)器號時(shí)

  • 有參構(gòu)造 開發(fā)者自行設(shè)置機(jī)器號

protected static long getDatacenterId(long maxDatacenterId) {
    long id = 0L;
    try {
        //獲取本機(jī)(或者服務(wù)器ip地址)
        //DESKTOP-123SDAD/192.168.1.87
        InetAddress ip = InetAddress.getLocalHost();
        NetworkInterface network = NetworkInterface.getByInetAddress(ip);
        //一般不是null會(huì)進(jìn)入else
        if (network == null) {
            id = 1L;
        } else {
            //獲取物理網(wǎng)卡地址
            byte[] mac = network.getHardwareAddress();
            if (null != mac) {
                id = ((0x000000FF & (long) mac[mac.length - 2]) | (0x0000FF00 & (((long) mac[mac.length - 1]) << 8))) >> 6;
                id = id % (maxDatacenterId + 1);
            }
        }
    } catch (Exception e) {
        logger.warn(" getDatacenterId: " + e.getMessage());
    }
    return id;
}
/**
 * 獲取 maxWorkerId
 */
protected static long getMaxWorkerId(long datacenterId, long maxWorkerId) {
    StringBuilder mpid = new StringBuilder();
    mpid.append(datacenterId);
    //獲取jvm進(jìn)程信息
    String name = ManagementFactory.getRuntimeMXBean().getName();
    if (StringUtils.isNotBlank(name)) {
        /*
         * 獲取進(jìn)程PID
         */
        mpid.append(name.split(StringPool.AT)[0]);
    }
    /*
     * MAC + PID 的 hashcode 獲取16個(gè)低位
     */
    return (mpid.toString().hashCode() & 0xffff) % (maxWorkerId + 1);
}

感謝你能夠認(rèn)真閱讀完這篇文章,希望小編分享的“mybatis-plus雪花算法自動(dòng)生成機(jī)器id原理的示例分析”這篇文章對大家有幫助,同時(shí)也希望大家多多支持億速云,關(guān)注億速云行業(yè)資訊頻道,更多相關(guān)知識等著你來學(xué)習(xí)!

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

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

AI