SpringBoot Redis 延时队列

本文探讨了如何使用Redis的有序集合ZSet实现延迟队列,优点包括灵活性、消息持久化、分布式支持和高可用性。尽管存在资源消耗、时间差问题和可能的重复消费,但可以通过策略优化解决这些问题。文章详细介绍了延迟消息的主体、工具类、生产者、消费者及消费实现。
摘要由CSDN通过智能技术生成

常见延迟队列实现方式

  延迟队列的实现方式有很多种,通过程序的方式实现,例如 JDK 自带的延迟队列 DelayQueue,通过 MQ 框架来实现,例如 RocketMQ、RabbitMQ等,本文通过 Redis 的方式来实现延迟队列 。
  Redis 是通过有序集合(ZSet)的方式来实现延迟消息队列的,ZSet 有一个 Score 属性可以用来存储延迟执行的时间。
  优点:
    灵活方便,Redis 是互联网公司的标配,无序额外搭建相关环境;
    可进行消息持久化,大大提高了延迟队列的可靠性;
    分布式支持,不像 JDK 自身的 DelayQueue;
    高可用性,利用 Redis 本身高可用方案,增加了系统健壮性。
  缺点:
    需要使用无限循环的方式来执行任务检查,会消耗少量的系统资源。
    redis的失效监听事件会存在一定的时间差,并且当数据量越大时,误差会越大。
    分布式业务的场景下,会出现重复消费的问题。(可以增加分布式锁的实现,但是redisson分布式锁提供了另一种延迟队列的实现方式)

延迟消息主体

/**
 * 延迟消息主体
 *
 * @author cheneq
 */
@Data
public class DelayMessage implements Serializable {
   


    /**
     * 消息队列组
     */
    private String groupName;

    /**
     * 消息id
     */
    private String id;

    /**
     * 消息体,对应业务内容
     */
    private Object body;

    /**
     * 消息过期时间 单位:秒
     */
    private long expireTime;


}

延迟消息工具类

/**
 * 延迟消息工具类
 *
 * @author cheneq
 */
@Component
public class DelayMessageUtils {
   

    @Resource
    private RedisTemplate<String, Object> redisTemplate;

    /**
     * 存入消息池
     *
     * @param delayMessage
     */
    public void addMsgPool(DelayMessage delayMessage) {
   
        if (null != delayMessage) {
   
            ValueOperations<String, Object> valueOperations = redisTemplate.opsForValue();
            String key = RedisKey.MESSAGE_POOL + delayMessage.getGroupName() + delayMessage.getId();
            valueOperations.set(key, JSON.toJSONString(delayMessage), delayMessage.getExpireTime(), TimeUnit.SECONDS);
        }
    }

    /**
     * 从消息池中删除消息
     *
     * @param groupName
     * @param id
     */
    public void delMsgPool(String groupName, String id) {
   
        String key = RedisKey.MESSAGE_POOL + groupName + id;
        redisTemplate.delete(key);
    }

    /**
     * 向队列中添加消息
     *
     * @param key
     * @param val
     * @param score 优先级
     * @return 返回消息id
     */
    public void addMsgQueue(String key, String val, long score) {
   
        ZSetOperations<String, Object> zSetOperations = redisTemplate.opsForZSet();
        zSetOperations.add(key, val, score);
    }

    /**
     * 从队列删除消息
     *
     * @param id
     * @return
     */
    
  • 1
    点赞
  • 5
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
Spring Boot提供了对Redis Stream队列的支持。Redis Stream是一个高效的持久化消息队列,可以用于实现发布-订阅模式、任务队列等应用场景。 要在Spring Boot中使用Redis Stream队列,需要进行以下步骤: 1. 添加RedisSpring Data Redis的依赖:在`pom.xml`文件中添加以下依赖: ```xml <dependency> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-starter-data-redis</artifactId> </dependency> ``` 2. 配置Redis连接信息:在`application.properties`或`application.yml`文件中配置Redis连接信息,包括主机、端口、密码等。 ```properties spring.redis.host=your_redis_host spring.redis.port=your_redis_port spring.redis.password=your_redis_password ``` 3. 创建Redis Stream队列:在Spring Boot中,可以使用`StreamOperations`接口来操作Redis Stream队列。可以通过自动注入`RedisTemplate`或`StringRedisTemplate`来获取`StreamOperations`对象。 ```java @Autowired private RedisTemplate<String, String> redisTemplate; ... StreamOperations<String, String, String> streamOperations = redisTemplate.opsForStream(); ``` 4. 发布消息到Redis Stream队列:使用`XADD`命令将消息发布到Redis Stream队列中。 ```java Map<String, String> message = new HashMap<>(); message.put("key1", "value1"); message.put("key2", "value2"); streamOperations.add("your_stream_key", message); ``` 5. 消费Redis Stream队列消息:使用`XREADGROUP`命令消费Redis Stream队列中的消息。 ```java Consumer<String, String> consumer = StreamOffset.create("your_stream_key", ReadOffset.lastConsumed()); while (true) { List<MapRecord<String, String, String>> records = streamOperations.read(consumer, StreamReadOptions.empty()); for (MapRecord<String, String, String> record : records) { // 处理消息 System.out.println(record.getValue()); } } ``` 以上是使用Spring Boot操作Redis Stream队列的基本步骤,你可以根据实际需求进行修改和扩展。希望对你有所帮助!
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值