【Redis实现全局唯一ID】

一、全局唯一ID的需求产生。

在订单业务中,我们需要保证id是绝对唯一的。

使用数据库自增长的id在分布式的情况下把表做了拆分处理后有可能会出现id重复的情况,这就违背了唯一性。而且数据自增长的id有很强的规律性,可以根据id推断出订单的数量信息,这也是不安全的。

二、使用redis实现全局唯一ID生成器。

redis可以用作全局唯一ID生成器实现的原因:

  • 递增性:redis本身就采用递增方案。
  • 唯一性:redis是独立于数据库之外的,不论数据库有几张表,redis只有一个,使用incr命令自增是唯一的。
  • 高可用性:redis可以用主从,集群等等方案可以确保它的高可用性。
  • 安全性:使用时间戳进行拼接确保安全性。并且Redis 的自增命令是原子性的,保证了在多线程或多进程并发情况下生成的 ID 的唯一性。
  • 高性能:Redis 是一个基于内存的数据库,读写速度非常快

    使用这种拼接方式,前面为31位的时间戳,大约为68年,理论完全可用,但是在高并发的情况下,每秒可能有很多个订单,这种方式支持2^32个订单,完全可用。

代码实现

java 复制代码
public class RedisIdWorker {
    /**
     * 开始时间戳
     */
    private static final long BEGIN_TIMESTAMP = 1640995200L;//2022年1月1日0时0分0秒对应的时间戳
    /**
     * 序列号的位数
     */
    private static final int COUNT_BITS = 32;

    private StringRedisTemplate stringRedisTemplate;

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

    public long nextId(String keyPrefix) {
        // 1.生成时间戳
        LocalDateTime now = LocalDateTime.now();
        long nowSecond = now.toEpochSecond(ZoneOffset.UTC);
        long timestamp = nowSecond - BEGIN_TIMESTAMP;

        // 2.生成序列号
        // 2.1.获取当前日期,精确到天
        String date = now.format(DateTimeFormatter.ofPattern("yyyy:MM:dd"));
        // 2.2.自增长
        long count = stringRedisTemplate.opsForValue().increment("icr:" + keyPrefix + ":" + date);

        // 3.拼接并返回
        return timestamp << COUNT_BITS | count;//使用逻辑运算符拼接
    }
}
相关推荐
农民也会写代码几秒前
dedecms织梦arclist标签noflag属性过滤多个参数
开发语言·数据库·sql·php·dedecms
m0_748232926 分钟前
你还在手动画ER图吗?让SQL自动生成ER图,轻松解决作业难题!
数据库·sql·oracle
清流君9 分钟前
【MySQL】数据库 Navicat 可视化工具与 MySQL 命令行基本操作
数据库·人工智能·笔记·mysql·ue5·数字孪生
邂逅岁月9 分钟前
MySQL表的增删改查初阶(下篇)
数据库·sql·mysql
道友老李9 分钟前
【存储中间件】Redis核心技术与实战(五):Redis缓存使用问题(BigKey、数据倾斜、Redis脑裂、多级缓存)、互联网大厂中的Redis
redis·缓存·中间件
Python_金钱豹10 分钟前
Text2SQL零代码实战!RAGFlow 实现自然语言转 SQL 的终极指南
前端·数据库·sql·安全·ui·langchain·机器人
静听夜半雨12 分钟前
CANoe入门——3、新建LIN工程及LIN DataBase(LDF文件)的创建
网络·数据库·c++·编辑器
DarkAthena1 小时前
【ORACLE】记录一些ORACLE的merge into语句的BUG
数据库·oracle·bug
大新新大浩浩2 小时前
arm64适配系列文章-第三章-arm64环境上mariadb的部署
数据库·arm·mariadb
聪明的墨菲特i2 小时前
SQL进阶知识:九、高级数据类型
xml·数据库·sql·mysql·json·空间数据类型