mybatis-plus 雪花算法id冲突问题解决、雪花算法id冲突、雪花算法、id冲突解决、id、id冲突、主键冲突

mybatis-plus 雪花算法id冲突问题解决、雪花算法id冲突、雪花算法、id冲突解决、id、id冲突、主键冲突

问题:
k8s里起了多个pod,发现mybatis-plus的雪花算法不同pod之前生成了相同的id

问题原因:
mybatis-plus默认id生成器生成datacenterId时是读取的机器网卡mac地址后两个字节,生成一个0~31的数字,这里有很大机率生成相同的datacenterId值
mybatis-plus默认id生成器生成workerId时是读取的jvm pid,但是k8s里pod的jvm pid不作处理时默认都是1,导致datacenterId + workerId有很大机率不同的pod拥有相同的值

分析日志:
ee:dc:39:3f:1a:53
531a >> 6 = 0101001100
0101001100 % 32 = 01100

86:65:49:0a:d1:f4
f4d1 >> 6 = 1111010011
1111010011 % 32 = 10011

e2:71:3a:cc:31:c3
c331 >> 6 = 1100001100
1100001100 % 32 = 01100

解决思路:
不同的pod之间分配唯一的datacenterId、workerId,使生成的雪花算法id不会产生冲突

解决方案(已实际使用并已通过测试):

//程序配置
application.yml:

mybatis-plus-ext:
  snowflake:
    redis-key: smarthome.mybatis-plus-ext.snowflake.redis-key #mybatis-plus 雪花算法生成器配置。解决多个pod情况下生成的 数据中心id+workerId 重复的问题的redis键值


 

//配置类
MybatisPlusSnowflakeConfig.java:

package com.onbright.smarthome.config;

import com.baomidou.mybatisplus.core.incrementer.DefaultIdentifierGenerator;
import com.baomidou.mybatisplus.core.incrementer.IdentifierGenerator;
import lombok.extern.slf4j.Slf4j;
import org.apache.commons.lang3.tuple.MutablePair;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.beans.factory.annotation.Value;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
import org.springframework.data.redis.core.RedisTemplate;

/**
 * <p>
 * mybatis-plus 雪花算法生成器配置。解决多个pod情况下生成的 数据中心id+workerId 重复的问题
 * </p>
 */
@Slf4j
@Configuration
public class MybatisPlusSnowflakeConfig {

    @Autowired
    private RedisTemplate redisTemplate;
    @Value("${mybatis-plus-ext.snowflake.redis-key}")
    private String redisKey;

    @Bean
    public IdentifierGenerator identifierGenerator(){
        //tested
        Long redisIncrementValue = redisTemplate.opsForValue().increment(redisKey, 1L);
        if (redisIncrementValue == null || redisIncrementValue < 0L){
            throw new RuntimeException(String.format("程序启动失败, 使用redis获取到的雪花算法索引值异常, %s", redisIncrementValue));
        }

        //tested
        //从redis索引值中取出dataCenterId、workerId
        MutablePair<Long, Long> longLongMutablePair = generateDatacenterIdAndWorkerIdFromIndex(redisIncrementValue);
        Long dataCenterId = longLongMutablePair.getLeft();
        Long workerId = longLongMutablePair.getRight();

        //tested
        //再次检查dataCenterId、workerId值
        if (dataCenterId == null || dataCenterId < 0L || dataCenterId > 31L
                || workerId == null || workerId < 0L || workerId > 31L
        ){
            throw new RuntimeException(String.format("程序启动失败, 使用雪花算法索引值算出的dataCenterId,workerId异常, %s, %s, %s", redisIncrementValue, dataCenterId, workerId));
        }

        log.info("mybatis-plus雪花算法datacenterId,workerId生成, redisIncrementValue: {}, dataCenterId: {}, workerId: {}", redisIncrementValue, dataCenterId, workerId);

        return new DefaultIdentifierGenerator(workerId, dataCenterId);
    }

    /**
     * 从index值中生成雪花算法的datacenterId(5bit)、workerId(5bit)
     *
     * @param index 索引值,如:1。必须为大于等于0的值
     * @return 返回生成好的用MutablePair包装的datacenterId(left)、workerId(right),其中datacenterId值范围为0~31,workerId值范围也为0~31,如:{0,0}、{1、1}、{0,1}、{0,31}、{31,31}
     * @throws IllegalArgumentException 当参数索引值小于0时会抛出异常
     */
    public static MutablePair<Long, Long> generateDatacenterIdAndWorkerIdFromIndex(long index) {
        if (index < 0) {
            throw new IllegalArgumentException("索引值不能为负数");
        }

        long indexBucket = index % 1024L;
        long dataCenterId = indexBucket / 32L;
        long workerId = indexBucket % 32L;

        return MutablePair.of(dataCenterId, workerId);
    }

}


//测试核心函数
MybatisPlusSnowflakeConfigTest.java:

package com.onbright.smarthome.config;

import org.apache.commons.lang3.tuple.MutablePair;
import org.junit.Assert;
import org.junit.Test;


public class MybatisPlusSnowflakeConfigTest {

    @Test
    public void generateDatacenterIdAndWorkerIdFromIndex() {
        //legal
        {
            MutablePair<Long, Long> longLongMutablePair = MybatisPlusSnowflakeConfig.generateDatacenterIdAndWorkerIdFromIndex(0L);
            Assert.assertEquals(0L, longLongMutablePair.getLeft().longValue());
            Assert.assertEquals(0L, longLongMutablePair.getRight().longValue());
        }
        {
            MutablePair<Long, Long> longLongMutablePair = MybatisPlusSnowflakeConfig.generateDatacenterIdAndWorkerIdFromIndex(1L);
            Assert.assertEquals(0L, longLongMutablePair.getLeft().longValue());
            Assert.assertEquals(1L, longLongMutablePair.getRight().longValue());
        }
        {
            MutablePair<Long, Long> longLongMutablePair = MybatisPlusSnowflakeConfig.generateDatacenterIdAndWorkerIdFromIndex(2L);
            Assert.assertEquals(0L, longLongMutablePair.getLeft().longValue());
            Assert.assertEquals(2L, longLongMutablePair.getRight().longValue());
        }
        {
            MutablePair<Long, Long> longLongMutablePair = MybatisPlusSnowflakeConfig.generateDatacenterIdAndWorkerIdFromIndex(31L);
            Assert.assertEquals(0L, longLongMutablePair.getLeft().longValue());
            Assert.assertEquals(31L, longLongMutablePair.getRight().longValue());
        }
        {
            MutablePair<Long, Long> longLongMutablePair = MybatisPlusSnowflakeConfig.generateDatacenterIdAndWorkerIdFromIndex(32L);
            Assert.assertEquals(1L, longLongMutablePair.getLeft().longValue());
            Assert.assertEquals(0L, longLongMutablePair.getRight().longValue());
        }
        {
            MutablePair<Long, Long> longLongMutablePair = MybatisPlusSnowflakeConfig.generateDatacenterIdAndWorkerIdFromIndex(33L);
            Assert.assertEquals(1L, longLongMutablePair.getLeft().longValue());
            Assert.assertEquals(1L, longLongMutablePair.getRight().longValue());
        }
        {
            MutablePair<Long, Long> longLongMutablePair = MybatisPlusSnowflakeConfig.generateDatacenterIdAndWorkerIdFromIndex(63L);
            Assert.assertEquals(1L, longLongMutablePair.getLeft().longValue());
            Assert.assertEquals(31L, longLongMutablePair.getRight().longValue());
        }
        {
            MutablePair<Long, Long> longLongMutablePair = MybatisPlusSnowflakeConfig.generateDatacenterIdAndWorkerIdFromIndex(64L);
            Assert.assertEquals(2L, longLongMutablePair.getLeft().longValue());
            Assert.assertEquals(0L, longLongMutablePair.getRight().longValue());
        }
        {
            MutablePair<Long, Long> longLongMutablePair = MybatisPlusSnowflakeConfig.generateDatacenterIdAndWorkerIdFromIndex(991L);
            Assert.assertEquals(30L, longLongMutablePair.getLeft().longValue());
            Assert.assertEquals(31L, longLongMutablePair.getRight().longValue());
        }
        {
            MutablePair<Long, Long> longLongMutablePair = MybatisPlusSnowflakeConfig.generateDatacenterIdAndWorkerIdFromIndex(992L);
            Assert.assertEquals(31L, longLongMutablePair.getLeft().longValue());
            Assert.assertEquals(0L, longLongMutablePair.getRight().longValue());
        }
        {
            MutablePair<Long, Long> longLongMutablePair = MybatisPlusSnowflakeConfig.generateDatacenterIdAndWorkerIdFromIndex(993L);
            Assert.assertEquals(31L, longLongMutablePair.getLeft().longValue());
            Assert.assertEquals(1L, longLongMutablePair.getRight().longValue());
        }
        {
            MutablePair<Long, Long> longLongMutablePair = MybatisPlusSnowflakeConfig.generateDatacenterIdAndWorkerIdFromIndex(1023L);
            Assert.assertEquals(31L, longLongMutablePair.getLeft().longValue());
            Assert.assertEquals(31L, longLongMutablePair.getRight().longValue());
        }
        {
            MutablePair<Long, Long> longLongMutablePair = MybatisPlusSnowflakeConfig.generateDatacenterIdAndWorkerIdFromIndex(1024L);
            Assert.assertEquals(0L, longLongMutablePair.getLeft().longValue());
            Assert.assertEquals(0L, longLongMutablePair.getRight().longValue());
        }

        //integer.max
        {
            MutablePair<Long, Long> longLongMutablePair = MybatisPlusSnowflakeConfig.generateDatacenterIdAndWorkerIdFromIndex(2147483647L);
            Assert.assertEquals(31L, longLongMutablePair.getLeft().longValue());
            Assert.assertEquals(31L, longLongMutablePair.getRight().longValue());
        }
        //long.max
        {
            MutablePair<Long, Long> longLongMutablePair = MybatisPlusSnowflakeConfig.generateDatacenterIdAndWorkerIdFromIndex(9223372036854775807L);
            Assert.assertEquals(31L, longLongMutablePair.getLeft().longValue());
            Assert.assertEquals(31L, longLongMutablePair.getRight().longValue());
        }

        //illegal
        {
            Assert.assertEquals("索引值不能为负数", Assert.assertThrows(IllegalArgumentException.class, () -> MybatisPlusSnowflakeConfig.generateDatacenterIdAndWorkerIdFromIndex(-1)).getMessage());
        }
    }
}

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值