雪花算法简单实现

雪花算法简单实现

Snowflake算法是一种分布式系统生成全局唯一ID的算法,主要用于在分布式系统中生成唯一ID,以防止因为多台服务器同时生成ID而产生重复的情况。其主要原理是根据时间戳、数据中心ID和机器ID等信息生成一个64位的唯一ID。

以下是Snowflake算法的主要框架Java源代码示例:

public class LeafIdGenerator {
    // 起始的时间戳
    private final long START_TIMESTAMP = 1614153600000L; // 2021-02-24 00:00:00
    // 每部分占用的位数
    private final long SEQUENCE_BITS = 12; // 序列号位数
    private final long MACHINE_ID_BITS = 5; // 机器id位数
    private final long DATACENTER_ID_BITS = 5; // 数据中心id位数
    // 每部分的最大值
    private final long MAX_MACHINE_ID = -1L ^ (-1L << MACHINE_ID_BITS);
    private final long MAX_DATACENTER_ID = -1L ^ (-1L << DATACENTER_ID_BITS);
    // 每部分向左的位移
    private final long MACHINE_ID_SHIFT = SEQUENCE_BITS; // 机器ID向左移动位数
    private final long DATACENTER_ID_SHIFT = SEQUENCE_BITS + MACHINE_ID_BITS; // 数据中心ID向左移动位数
    private final long TIMESTAMP_SHIFT = SEQUENCE_BITS + MACHINE_ID_BITS + DATACENTER_ID_BITS; // 时间戳向左移动位数
    // 数据中心ID和机器ID
    private long datacenterId;
    private long machineId;
    // 序列号
    private long sequence = 0L;
    private long lastTimestamp = -1L;

    public LeafIdGenerator(long datacenterId, long machineId) {
        if (datacenterId > MAX_DATACENTER_ID || datacenterId < 0) {
            throw new IllegalArgumentException("Datacenter ID can't be greater than " + MAX_DATACENTER_ID + " or less than 0");
        }
        if (machineId > MAX_MACHINE_ID || machineId < 0) {
            throw new IllegalArgumentException("Machine ID can't be greater than " + MAX_MACHINE_ID + " or less than 0");
        }
        this.datacenterId = datacenterId;
        this.machineId = machineId;
    }

    public synchronized long generateId() {
        long timestamp = System.currentTimeMillis();

        if (timestamp < lastTimestamp) {
            throw new RuntimeException("Clock moved backwards. Refusing to generate id");
        }

        if (timestamp == lastTimestamp) {
            sequence = (sequence + 1) & ((1 << SEQUENCE_BITS) - 1);
            if (sequence == 0) {
                timestamp = tilNextMillis(lastTimestamp);
            }
        } else {
            sequence = 0;
        }

        lastTimestamp = timestamp;

        return ((timestamp - START_TIMESTAMP) << TIMESTAMP_SHIFT)
                | (datacenterId << DATACENTER_ID_SHIFT)
                | (machineId << MACHINE_ID_SHIFT)
                | sequence;
    }

    private long tilNextMillis(long lastTimestamp) {
        long timestamp = System.currentTimeMillis();
        while (timestamp <= lastTimestamp) {
            timestamp = System.currentTimeMillis();
        }
        return timestamp;
    }

    public static void main(String[] args) {
        LeafIdGenerator idGenerator = new LeafIdGenerator(1, 1);

        for (int i = 0; i < 10; i++) {
            long id = idGenerator.generateId();
            System.out.println("Generated ID: " + id);
        }
    }
}
  • 9
    点赞
  • 7
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

semicolon_helloword

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值