雪花算法-java实现

本文详细介绍了雪花算法的原理及其在分布式环境中生成唯一ID的作用。通过64位二进制划分,包括时间戳、机器标识、序列号等部分,确保了ID的全局唯一性。同时,文章讨论了时间回拨问题的解决方案,并提供了Java实现雪花算法的代码示例,展示了如何处理并发和序列号重置。此外,还提到了序列号初始化可能导致的偶数ID问题及其解决方案。
摘要由CSDN通过智能技术生成

雪花算法用途

用来保证分布式环境生成ID唯一。

雪花算法实现

雪花算法用64位二进制表示,其中二进制位数作用划分为:

  • 1bit:符号位,无使用
  • 41bit:时间位,用于表示毫秒可以使用69年
  • 10bit:机器位,用来区分不同的机器环境
  • 12bit:序列位,用来表示同一毫秒的不同序列,同一毫秒的并发数。
    上面位数划分不是固定的,可以自定义划分,如:
  • 1bit:符号位,无使用
  • 41bit:时间位,用于表示毫秒
  • 5bit:机器位
  • 7bit:业务位,用来区分不同业务
  • 10bit:序列位

雪花算法实现注意事项

  • 每毫秒的序列号都从0开始的话,会导致没有竞争情况返回的都是偶数。解决方法是用时间戳&1,这样就会随机得到1或者0。
  • 时间回拨问题。生成id时得到的时间戳是更早些时候的,这样可能会导致生成重复的id。其中简单的解决方法是直接抛出异常。

java实现

public class SnowService {
    /**
     * 机器占用位数,范围是0-63
     */
    private long machineBit = 5L;
    /**
     * 业务占用位数,范围0-255
     */
    private long businessBit = 7L;
    /**
     * 序列占用位数,范围0-2047
     */
    private long sequenceBit = 10L;
    /**
     * 传入的机器数值
     */
    private long machineNo;
    /**
     * 传入业务数值
     */
    private long businessNo;
    /**
     * 同一毫秒序列号
     */
    private long sequenceNo;
    /**
     * 最后获取id的时间戳
     */
    private long lastTimeStamp = -1L;
    /**
     * 时间位需要左移的位数
     * 等于机器位数+业务位数+序列位数
     * 即等于低位的总位数
     */
    private long timeStampShift = machineBit + businessBit + sequenceBit;
    /**
     * 机器位需要左移的位数
     * 等于业务位+序列位
     */
    private long machineShift = businessBit + sequenceBit;
    /**
     * 业务位需要左移的位数
     * 等于序列位
     */
    private long businessShift = sequenceBit;
    /**
     * sequence的最大值
     */
    private long maxSequence = -1L ^ (-1L << sequenceBit);
    /**
     * 开始使用时间
     */
    private long startStamp = 1624348225278L;
    public SnowService(long machineNo, long businessNo) {
        this.machineNo = machineNo;
        this.businessNo = businessNo;
    }

    public synchronized long getId() {
        long nowStamp = System.currentTimeMillis();
        //当前时间小于上次处理时间,表示时间回拨,直接抛出异常
        if(nowStamp < lastTimeStamp) {
            throw new RuntimeException("时间小于上次获取id时间");
        }
        if(nowStamp == lastTimeStamp) {
            //超过最大值把sequenceNo重置为0
            sequenceNo = (sequenceNo + 1) & maxSequence;
            if(sequenceNo == 0) {
                nowStamp = waitNextMillis(nowStamp);
            }
        } else {
            sequenceNo = nowStamp & 1;
            //sequenceNo =0;这种每毫秒都从0开始会导致如果没有冲突生成的id都是偶数
        }
        lastTimeStamp = nowStamp;
        long id = (nowStamp - startStamp) << timeStampShift |
                machineNo << machineShift |
                businessNo << businessShift |
                sequenceNo;
        return id;
    }

    /**
     * 等待到下一毫秒
     * @param nowStamp
     * @return
     */
    private long waitNextMillis(long nowStamp) {
        long newStamp = System.currentTimeMillis();
        while(nowStamp >= newStamp) {
            newStamp = System.currentTimeMillis();
        }
        return newStamp;
    }

    public static void main(String[] args) throws InterruptedException {
        SnowService snowService = new SnowService(1, 1);
        for(int i = 0; i < 100; i++) {
            Thread t = new Thread(() -> {
                System.out.println(snowService.getId());
            });
            t.start();
        }
    }
}
  • 2
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
以下是雪花算法Java实现: ```java public class SnowFlake { private long workerId; private long datacenterId; private long sequence = 0L; private long twepoch = 1288834974657L; private long workerIdBits = 5L; private long datacenterIdBits = 5L; private long maxWorkerId = -1L ^ (-1L << workerIdBits); private long maxDatacenterId = -1L ^ (-1L << datacenterIdBits); private long sequenceBits = 12L; private long workerIdShift = sequenceBits; private long datacenterIdShift = sequenceBits + workerIdBits; private long timestampLeftShift = sequenceBits + workerIdBits + datacenterIdBits; private long sequenceMask = -1L ^ (-1L << sequenceBits); private long lastTimestamp = -1L; public SnowFlake(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(); } } ```
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值