redis工具集之基于zset设计的延迟队列方案

代码仓库

gitee: gitee.com/listen_w/re...

github: github.com/jettwangcj/...

背景

我司现有业务中有很多场景需要使用延迟队列,之前开发了一个基于 redisson + kafka的延迟队列设计方案,设计思路参考博文基于 Redisson 和 Kafka 的延迟队列设计方案

但是有同学使用过程中发现这个延迟队列组件有BUG,具体是消息丢失。

所以,需要有其他延迟队列方案来替换,目前调研之后有以下两个思路:

  1. 接入新的支持延迟队列消息的消息中间件,例如: Rocketmq。
  2. 自研一个延迟队列组件,例如:基于 ZSET 实现。

使用方式

  1. 引入依赖

    xml 复制代码
    <dependency>
        <groupId>cn.org.wangchangjiu</groupId>
        <artifactId>redis-util-spring-boot-starter</artifactId>
        <version>1.0.1-SNAPSHOT</version>
    </dependency>
  2. 开启配置

  3. 发送延迟消息

  4. 接收延迟消息

    1. 自动ACK,方法上增加 @RedisDelayMessageListener 注解,topic 和发送的topic对应,值得注意的事,自动ACK只能是一个参数,参数就是所需要接收的消息实体。

    2. 手动ACK,方法上增加 @RedisDelayMessageListener 注解,topic 和发送的topic对应,值得注意的事,手动ACK必须是两个参数,第一个参数是所需要接收的消息实体,第二个参数固定是 Acknowledgment,当执行完方法是或者有需要时通过 acknowledgment.acknowledge() 手动提交ACK,如果未提交ACK,那么过一段时间(时间可配置),该消息会被重新投递到延迟队列中。

设计方案流程图

重点流程说明

  1. 服务启动后会启动三个线程组:boss 线程池、worker线程池、back线程池;

    Boss 线程每隔500毫秒(可配置)周期性发布获取延迟消息的任务;

    worker线程根据不同的topic去获取延迟消息,如果获取到延迟消息,那么反射调用目标方法执行,执行成功自动/手动ACK消息;

    back线程每隔5秒(可配置)周期性检查是否有未提交ACK的消息,如果有,则重新投递到延迟队列中;

  2. 服务启动后,boss 线程会发布批量获取延迟消息的任务,worker线程执行任务,批量获取延迟消息,这么做的原因是,怕服务宕机或者重启时,大量延迟消息未消费,所以一次性批量消费。

  3. 发送延迟消息过程:

    • 添加消息元数据,元数据存储在 hash数据结构中,key : task_meta_data:{service} (service为服务名称)

    • 将消息ID添加到 ZSET数据结构中,value为MessageID,score为 当前时间戳+过期时间

实现细节

整体概论

核心Bean 简介

DelayQueueAutoConfiguration:延迟队列自动装配Bean。

DelayQueueMessageProducer:延迟消息发送器。

DelayQueueMessageConsumer延迟消息消费者,该Bean实现了 BeanPostProcessor 接口,主要作用是,获取被注解 @RedisMessageListener 修饰的方法,把信息封装在 RedisDelayMessageConsumerContainer对象里,方便后面反射调用。

DelayQueue:延迟队列,发送消息/获取消息/提交 ACK 核心处理类,该类封装了 redis + lua的脚本,通过调用lua脚本处理redis延迟消息;

服务启动以及发送消息调用过程

以上流程介绍

  1. 容器启动时主要操作如下:

    • 检查哪些Bean的方法上有注解@RedisDelayMessageListener标注的,封装目标方法信息,用于反射调用;如下图:会将以下方法信息封装到 RedisDelayMessageConsumerContainer 对象里。 实现代码:

    • 启动三个线程组,boss 线程池、worker线程池、back线程池

      • Boss 线程每隔500毫秒(可配置)周期性发布获取延迟消息的任务;

      • worker线程根据不同的topic去获取延迟消息,如果获取到延迟消息,那么反射调用目标方法执行,执行成功自动/手动ACK消息;

      • back线程每隔5秒(可配置)周期性检查是否有未提交ACK的消息,如果有,则重新投递到延迟队列中;

  2. 发送消息

    生成消息ID(UUID),调用lua脚本添加消息

  3. 获取消息

  1. 检查ACK

不足之处

当没有延迟消息,或者延迟消息延迟时间过长时,会导致很多空轮询检查消息,浪费性能。

推荐环节

推荐自己开发的两款插件:

  1. SQLToMongo 这是一款能将SQL查询语句翻译成Mongo语法的插件;

  2. Table2Entity 这是一款代码生成插件,通过建表语句可以生成jpa实体和repository,以及生成Mybatis实体和mapper;

源码

gitee: gitee.com/listen_w/pr...

github: github.com/jettwangcj?...

相关推荐
3***g2051 小时前
redis连接服务
数据库·redis·bootstrap
雪域迷影1 小时前
MacOS中安装并配置Redis
redis·macos·策略模式
雪域迷影1 小时前
Node.js中使用node-redis库连接redis服务端并存储数据
数据库·redis·node.js
雪域迷影2 小时前
Windows11上安装Redis服务和Redis可视化客户端
windows·redis
青云交2 小时前
Java 大视界 -- 基于 Java+Redis Cluster 构建分布式缓存系统:实战与一致性保障(444)
java·redis·缓存·缓存穿透·分布式缓存·一致性保障·java+redis clus
三不原则2 小时前
故障案例:模型推理响应慢,排查 Redis 缓存集群问题
数据库·redis·缓存
小北方城市网15 小时前
分布式锁实战指南:从选型到落地,避开 90% 的坑
java·数据库·redis·分布式·python·缓存
ohoy16 小时前
RedisTemplate 使用之Zset
java·开发语言·redis
冰冰菜的扣jio19 小时前
Redis缓存中三大问题——穿透、击穿、雪崩
java·redis·缓存
阿里巴巴P8资深技术专家20 小时前
基于 Spring AI 和 Redis 向量库的智能对话系统实践
人工智能·redis·spring