当前位置: 首页 > article >正文

Redis - 全局ID生成器 RedisIdWorker

文章目录

  • Redis - 全局ID生成器 RedisIdWorker
    • 一、引言
    • 二、实现原理
    • 三、代码实现
      • 代码说明
    • 四、使用示例
      • 示例说明
    • 五、总结

Redis - 全局ID生成器 RedisIdWorker

在这里插入图片描述

一、引言

在分布式系统中,生成全局唯一ID是一个常见的需求。传统的自增ID生成方式在分布式环境下容易出现冲突,而UUID虽然可以保证唯一性,但长度较长且不够紧凑。RedisIdWorker是一种基于Redis实现的全局ID生成器,它结合了时间戳和自增序列号,能够在分布式环境中高效地生成唯一且有序的ID。

二、实现原理

RedisIdWorker的核心思想是利用Redis的自增特性和时间戳来生成唯一且有序的ID。其生成的ID由两部分组成:

  1. 时间戳部分:使用当前时间戳减去一个起始时间戳(例如某个特定日期的时间戳),得到一个相对时间戳。时间戳部分通常占用31位,以秒为单位,这样可以保证在69年内生成的ID是唯一的。
  2. 自增序列号部分:使用Redis的INCR命令生成一个自增序列号,确保在相同的时间戳下,ID是唯一的。序列号部分通常占用32位,这意味着每秒可以生成2^32个不同的ID。

在具体实现中,时间戳部分和自增序列号部分通过位运算组合在一起。时间戳部分左移32位,然后与序列号部分进行按位或操作,最终生成一个64位的全局唯一ID。这种设计不仅保证了ID的唯一性,还确保了ID的递增性,有利于数据库索引的创建。

这种实现方式充分利用了Redis的原子操作特性,确保在高并发环境下生成的ID仍然是唯一的。同时,由于时间戳和序列号的结合,生成的ID具有一定的规律性,但又不会直接暴露业务逻辑。

三、代码实现

以下是RedisIdWorker的Java代码实现:

java复制

@Component
public class RedisIdWorker {
    // 开始时间戳(例如2022年1月1日)
    private static final long BEGIN_TIMESTAMP = 1640995200L;
    // 序列号的位数
    private static final int COUNT_BITS = 32;

    private StringRedisTemplate stringRedisTemplate;

    public RedisIdWorker(StringRedisTemplate stringRedisTemplate) {
        this.stringRedisTemplate = stringRedisTemplate;
    }

    // 获取下一个自动生成的ID
    public long nextId(String keyPrefix) {
        // 1. 生成时间戳
        LocalDateTime now = LocalDateTime.now();
        long nowSecond = now.toEpochSecond(ZoneOffset.UTC);
        long timestamp = nowSecond - BEGIN_TIMESTAMP;

        // 2. 获取当前日期,用于生成key
        String date = now.format(DateTimeFormatter.ofPattern("yyyy:MM:dd"));

        // 3. 获取自增序列号
        long count = stringRedisTemplate.opsForValue().increment("incr:" + keyPrefix + ":" + date);

        // 4. 拼接并返回ID
        return timestamp << COUNT_BITS | count;
    }
}

代码说明

  1. 时间戳部分:通过LocalDateTime获取当前时间戳,并减去起始时间戳BEGIN_TIMESTAMP
  2. 自增序列号部分:使用StringRedisTemplateincrement方法,为每个日期生成一个自增序列号。
  3. ID拼接:将时间戳左移32位,然后与序列号进行按位或操作,生成最终的ID。

四、使用示例

以下是一个简单的使用示例:

java复制

@SpringBootTest
public class RedisIdWorkerTest {
    @Resource
    private RedisIdWorker redisIdWorker;

    @Test
    public void testIdWorker() {
        // 生成订单ID
        long orderId = redisIdWorker.nextId("order");
        System.out.println("Generated Order ID: " + orderId);

        // 生成用户ID
        long userId = redisIdWorker.nextId("user");
        System.out.println("Generated User ID: " + userId);
    }
}

示例说明

  • nextId方法接收一个keyPrefix参数,用于区分不同类型的ID(例如订单ID、用户ID等)。
  • 每次调用nextId方法都会生成一个唯一的ID,并且由于时间戳和自增序列号的结合,生成的ID是严格递增的。

五、总结

RedisIdWorker是一种简单高效的全局ID生成器,特别适用于分布式系统。它通过结合时间戳和自增序列号,利用Redis的原子操作保证了ID的唯一性和有序性。在实际项目中,可以根据业务需求调整时间戳的起始值和序列号的位数,以满足不同的场景。


版权声明:本博客内容为原创,转载请保留原文链接及作者信息。

参考文章

  • [Redis - 全局ID生成器 RedisIdWorker - CSDN博客]
  • [Redis - 全局ID生成器 RedisIdWorker本文介绍了分布式系统中的全局ID生成器RedisIdWorke - 掘金]

http://www.kler.cn/a/535542.html

相关文章:

  • Qt:Qt环境配置安装
  • Python爬虫-如何正确解决起点中文网的无限debugger
  • 3. k8s二进制集群之负载均衡器高可用部署
  • 【初/高中生讲机器学习】0. 本专栏 “食用” 指南——写在一周年之际⭐
  • Java高频面试之SE-17
  • 缓存类为啥使用 unordered_map 而不是 map
  • TypeScript+React+Redux:类型安全的状态管理最佳实践
  • MySQL知识大总结(进阶)
  • 如何开设一个Facebook账户:详细步骤与注意事项
  • 人工智能丨利用人工智能与自动化实现高效运营推广
  • 十. Redis 事务和 “锁机制”——> 并发秒杀处理的详细说明
  • python爬虫常用库
  • 深入浅出 NVM:如何管理 Node 版本?
  • 8.[网鼎杯 2020 青龙组]AreUSerialz
  • DeepSeek技术报告解析:为什么DeepSeek-R1 可以用低成本训练出高效的模型
  • Beans模块之工厂模块注解模块InitDestroyAnnotationBeanPostProcessor
  • PostgreSQL存储过程和执行
  • 【工具篇】深度揭秘 Midjourney:开启 AI 图像创作新时代
  • 备战蓝桥杯-洛谷
  • 使用Python的Tabulate库优雅地格式化表格数据
  • SpringCloud Gateway 动态路由配置全解
  • 如何用大语言模型提高工作效率
  • fio使用手册
  • springboot005-Java沉浸式戏曲文化体验系统
  • Vue基础:计算属性(描述依赖响应式状态的复杂逻辑)
  • 使用Docker安装MongoDB数据库