基于Redission实现一个延迟队列的实践

举报
Jack20 发表于 2025/05/29 14:27:03 2025/05/29
【摘要】 基于 Redisson 实现延迟队列可以利用其内置的 RDelayedQueue 组件。以下是详细实现步骤和代码示例:​​1. 核心原理​​Redisson 的延迟队列基于 Redis 的有序集合(Sorted Set)和发布订阅(Pub/Sub)机制实现:​​有序集合​​:存储延迟元素,以到期时间作为分数(score)。​​后台线程​​:定期轮询有序集合,将到期元素转移到普通队列。​​消费...

基于 Redisson 实现延迟队列可以利用其内置的 RDelayedQueue 组件。以下是详细实现步骤和代码示例:


​1. 核心原理​

Redisson 的延迟队列基于 Redis 的有序集合(Sorted Set)和发布订阅(Pub/Sub)机制实现:

  • ​有序集合​​:存储延迟元素,以到期时间作为分数(score)。
  • ​后台线程​​:定期轮询有序集合,将到期元素转移到普通队列。
  • ​消费者​​:从普通队列中获取到期的消息。

​2. 实现步骤​

​2.1 添加依赖​

在 Maven 项目中添加 Redisson 依赖:

<dependency>
    <groupId>org.redisson</groupId>
    <artifactId>redisson</artifactId>
    <version>3.21.0</version> <!-- 使用最新版本 -->
</dependency>

​2.2 配置 Redisson 客户端​

import org.redisson.Redisson;
import org.redisson.config.Config;

public class RedissonConfig {
    public static RedissonClient getClient() {
        Config config = new Config();
        config.useSingleServer().setAddress("redis://127.0.0.1:6379");
        return Redisson.create(config);
    }
}

​2.3 创建延迟队列​

import org.redisson.api.RBlockingQueue;
import org.redisson.api.RDelayedQueue;
import org.redisson.api.RedissonClient;

public class DelayQueueExample {
    public static void main(String[] args) {
        RedissonClient redisson = RedissonConfig.getClient();
        
        // 普通阻塞队列(用于存放到期消息)
        RBlockingQueue<String> destinationQueue = redisson.getBlockingQueue("delayedQueue");
        
        // 延迟队列(绑定普通队列和延迟时间)
        RDelayedQueue<String> delayedQueue = new RDelayedQueue<>(redisson.getQueue("delayedQueue"), destinationQueue, 0, TimeUnit.SECONDS);
        
        // 生产者:发送延迟消息
        new Thread(() -> {
            try {
                delayedQueue.offer("Order123", 10, TimeUnit.SECONDS); // 10秒后到期
                System.out.println("Message sent with 10s delay");
            } catch (InterruptedException e) {
                Thread.currentThread().interrupt();
            }
        }).start();

        // 消费者:从目标队列获取到期消息
        new Thread(() -> {
            while (true) {
                try {
                    String message = destinationQueue.take();
                    System.out.println("Received: " + message);
                } catch (InterruptedException e) {
                    Thread.currentThread().interrupt();
                }
            }
        }).start();
    }
}

​3. 关键参数说明​

  • offer(element, delay, timeUnit)
    将元素加入延迟队列,delay 表示延迟时间,timeUnit 是时间单位(如秒、毫秒)。

  • take()
    阻塞式获取到期消息,若队列为空则等待。


​4. 高级用法​

​4.1 自定义消息对象​

public class OrderMessage implements Serializable {
    private String orderId;
    private long expireTime;
    // getters/setters
}

// 生产者发送
RDelayedQueue<OrderMessage> delayedQueue = ...;
delayedQueue.offer(new OrderMessage("Order123", System.currentTimeMillis() + 10_000), 0, TimeUnit.SECONDS);

// 消费者解析
OrderMessage msg = destinationQueue.take();

​4.2 多消费者并发处理​

// 使用线程池消费
ExecutorService executor = Executors.newFixedThreadPool(4);
for (int i = 0; i < 4; i++) {
    executor.submit(() -> {
        while (true) {
            String message = destinationQueue.take();
            processMessage(message);
        }
    });
}

​5. 注意事项​

  1. ​可靠性保证​

    • Redisson 内部通过定时任务轮询有序集合,确保消息到期后转移到目标队列。
    • 若 Redis 宕机,需结合持久化机制(如 RDB/AOF)保证数据不丢失。
  2. ​性能优化​

    • 避免在消费者中使用阻塞操作,防止线程耗尽。
    • 对于海量消息,建议使用 RPriorityQueue 或结合 RocketMQ 等专业消息队列。
  3. ​超时时间精度​
    Redisson 默认的轮询间隔是 5 秒,因此延迟时间精度为 ±5 秒。可通过修改配置调整:

    Config config = new Config();
    config.useSingleServer().setAddress("redis://127.0.0.1:6379");
    config.setScanInterval(2000); // 轮询间隔设为2秒(默认5秒)

​6. 完整流程图​

生产者调用 delayedQueue.offer(message, delay)
  → 元素存入 Redis Sorted Set(以到期时间作为 score)
  → Redisson 后台线程定期扫描 Sorted Set
  → 到期元素被移动到普通队列(destinationQueue)
  → 消费者通过 destinationQueue.take() 获取消息

通过以上步骤,你可以快速实现一个高可用的延迟队列。如果需要更复杂的调度(如动态调整延迟时间),可以结合 Lua 脚本或 Redis 的 ZREMRANGEBYSCORE 命令自行扩展。

【声明】本内容来自华为云开发者社区博主,不代表华为云及华为云开发者社区的观点和立场。转载时必须标注文章的来源(华为云社区)、文章链接、文章作者等基本信息,否则作者和本社区有权追究责任。如果您发现本社区中有涉嫌抄袭的内容,欢迎发送邮件进行举报,并提供相关证据,一经查实,本社区将立刻删除涉嫌侵权内容,举报邮箱: cloudbbs@huaweicloud.com
  • 点赞
  • 收藏
  • 关注作者

评论(0

0/1000
抱歉,系统识别当前为高风险访问,暂不支持该操作

全部回复

上滑加载中

设置昵称

在此一键设置昵称,即可参与社区互动!

*长度不超过10个汉字或20个英文字符,设置后3个月内不可修改。

*长度不超过10个汉字或20个英文字符,设置后3个月内不可修改。