使用 Redis + Redisson 分布式锁来生成全局唯一、线程安全的带日期前缀的流水号的完整实现。

业务需求:创建一个流水号,

功能说明:

1,格式:orgCode + yyyyMMdd + 4位序号,

2,每天重新从 0001 开始

3,使用 Redisson 保证并发下分布式唯一,

4,Redis 存储当天序列号

示例代码

  1. Maven 依赖(如果还没加 Redisson):
java 复制代码
<dependency>
    <groupId>org.redisson</groupId>
    <artifactId>redisson</artifactId>
    <version>3.17.7</version>
</dependency>
  1. 配置 RedissonClient(示例单节点):
java 复制代码
@Configuration
public class RedissonConfig {
    @Bean
    public RedissonClient redissonClient() {
        Config config = new Config();
        config.useSingleServer()
              .setAddress("redis://127.0.0.1:6379") // 修改为你自己的 Redis 地址
              .setDatabase(0);
        return Redisson.create(config);
    }
}
  1. 流水号工具类:
java 复制代码
@Component
public class SerialNumberGenerator {

    @Autowired
    private RedissonClient redissonClient;

    private static final String SERIAL_KEY_PREFIX = "serial:";

    public String generateSerialNumber(String orgCode) {
        String currentDate = new SimpleDateFormat("yyyyMMdd").format(new Date());
        String key = SERIAL_KEY_PREFIX + currentDate;

        String lockKey = "lock:serial:" + currentDate;
        RLock lock = redissonClient.getLock(lockKey);

        try {
            // 获取锁,最多等待5秒,锁自动释放时间为10秒
            if (lock.tryLock(5, 10, TimeUnit.SECONDS)) {
                RAtomicLong counter = redissonClient.getAtomicLong(key);

                // 如果是新的一天,重置为0
                if (!counter.isExists()) {
                    counter.set(0);
                    // 设置过期时间为2天,避免key一直存在
                    counter.expire(2, TimeUnit.DAYS);
                }

                long serial = counter.incrementAndGet();
                String formattedSerial = String.format("%04d", serial);
                return orgCode + currentDate + formattedSerial;
            } else {
                throw new RuntimeException("生成流水号失败:获取锁超时");
            }
        } catch (InterruptedException e) {
            throw new RuntimeException("生成流水号失败:线程中断", e);
        } finally {
            if (lock.isHeldByCurrentThread()) {
                lock.unlock();
            }
        }
    }
}

为什么需要 Redis 分布式锁?
不加锁的问题

Redis 是高并发环境下的共享存储,多个实例并发 get + set 操作有可能导致并发写丢失或重复编号。
加锁保证 :每次生成编号时只有一个线程在操作该 key 的值,从而保证原子性和唯一性

相关推荐
Java永无止境8 分钟前
延时任务之Redis 过期事件监听原理与缺陷
数据库·redis·缓存·延时任务
Albert Edison27 分钟前
【MySQL】表的操作
数据库·mysql·oracle
欢喜躲在眉梢里27 分钟前
mysql中的日志
android·运维·数据库·mysql·adb·日志·mysql日志
java1234_小锋36 分钟前
Redis线上操作最佳实践有哪些?
数据库·redis·缓存
合作小小程序员小小店1 小时前
web开发,在线%校园,论坛,社交管理%系统,基于html,css,python,django,mysql
数据库·后端·mysql·django·web app
失散131 小时前
分布式专题——22 Kafka集群工作机制详解
java·分布式·云原生·架构·kafka
清风6666662 小时前
基于51单片机的空气质量检测PM2.5粉尘检测设计
数据库·单片机·嵌入式硬件·毕业设计·51单片机·课程设计
Lansonli2 小时前
大数据Spark(六十五):Transformation转换算子groupByKey和filter
大数据·分布式·spark
倔强的石头_3 小时前
【金仓数据库】ksql 指南(一) 连接本地 KingbaseES 数据库与基础交互
数据库
拾忆,想起3 小时前
RabbitMQ死信交换机:消息的“流放之地“
开发语言·网络·分布式·后端·性能优化·rabbitmq