redisson实现延迟队列

1.redisson延迟队列工具类

java 复制代码
package com.cloud.app.system.utils;

import lombok.RequiredArgsConstructor;
import lombok.extern.slf4j.Slf4j;
import org.redisson.api.RDelayedQueue;
import org.redisson.api.RedissonClient;
import org.springframework.stereotype.Component;

import java.util.concurrent.TimeUnit;

@Slf4j
@Component
@RequiredArgsConstructor
public class RedissonDelayQueueUtil {

    private final RedissonClient redissonClient;

    /**
     * 获取延迟队列实例
     * 注意:RDelayedQueue 底层依赖一个普通的 BlockingDeque 来存储到期任务
     * queueName 是队列的唯一标识
     */
    private <T> RDelayedQueue<T> getDelayedQueue(String queueName) {
        return redissonClient.getDelayedQueue(redissonClient.getBlockingDeque(queueName));
    }

    /**
     * 发送延迟任务
     *
     * @param queueKey 队列名称 (例如: "order:timeout:queue")
     * @param task     任务对象 (必须可序列化,建议实现 Serializable 或使用 JSON 字符串)
     * @param delay    延迟时间数值
     * @param unit     时间单位
     */
    public <T> void sendTask(String queueKey, T task, long delay, TimeUnit unit) {
        try {
            RDelayedQueue<T> delayedQueue = getDelayedQueue(queueKey);
            delayedQueue.offer(task, delay, unit);
            log.debug("任务已加入 Redisson 延迟队列: {}, 延迟: {} {}", queueKey, delay, unit);
        } catch (Exception e) {
            log.error("发送延迟任务失败: {}", queueKey, e);
            throw new RuntimeException("发送延迟任务失败", e);
        }
    }

    /**
     * 取消任务 (在任务执行前移除)
     * 注意:这只能移除尚未转移到"就绪队列"的任务。
     * 如果任务已经到期并转移到了 BlockingDeque 中,则需要从 BlockingDeque 中移除,逻辑较复杂。
     * 通常延迟队列场景下,取消操作较少,或者接受任务执行时再判断状态。
     */
    public <T> boolean cancelTask(String queueKey, T task) {
        try {
            RDelayedQueue<T> delayedQueue = getDelayedQueue(queueKey);
            return delayedQueue.remove(task);
        } catch (Exception e) {
            log.error("取消任务失败", e);
            return false;
        }
    }

    /**
     * 获取关联的就绪队列 (用于消费者监听)
     * 当延迟时间到达后,任务会自动从 DelayedQueue 移动到这里的 BlockingDeque
     */
    public <T> org.redisson.api.RBlockingDeque<T> getReadyQueue(String queueKey) {
        return redissonClient.getBlockingDeque(queueKey);
    }
}

2.延迟队列生产者

java 复制代码
package com.cloud.app.system.queues.produce;


import com.cloud.app.system.config.BackupConfig;
import com.cloud.app.system.queues.ResourceBackUpTask;
import com.cloud.app.system.utils.RedissonDelayQueueUtil;
import lombok.extern.slf4j.Slf4j;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.stereotype.Component;

import java.util.concurrent.TimeUnit;

@Slf4j
@Component
public class ResourceBackUpProduce {

    @Autowired
    private BackupConfig backupConfig;

    @Autowired
    private  RedissonDelayQueueUtil delayQueueUtil;

    public static final String RESOURCE_CANCEL_QUEUE ="cloud:resource:backup:queue";


    public void createOrder(Long instanceId, Long userId, Long backUpId) {
        ResourceBackUpTask task = new ResourceBackUpTask(instanceId, userId, backUpId);
        delayQueueUtil.sendTask(RESOURCE_CANCEL_QUEUE, task, backupConfig.getBackUpTime(), TimeUnit.SECONDS);
        log.info("已加入延时队列,id为:{}",instanceId);
    }

}

3.延迟队列消费者

java 复制代码
package com.cloud.app.system.queues.consumer;


import com.cloud.app.common.mcclient.domain.CloudPhoneActionDataInfo;
import com.cloud.app.system.domain.CpAppInstanceBackInfo;
import com.cloud.app.system.domain.CpDiskInfo;
import com.cloud.app.system.domain.CpInstanceInfo;
import com.cloud.app.system.domain.vo.CpBackUpVo;
import com.cloud.app.system.mapper.CpAppInstanceBackInfoMapper;
import com.cloud.app.system.mapper.CpDiskInfoMapper;
import com.cloud.app.system.mapper.CpInstanceInfoMapper;
import com.cloud.app.system.queues.ResourceBackUpTask;
import com.cloud.app.system.service.ICpAppInstanceBackInfoService;
import com.cloud.app.system.service.ICpInstanceInfoService;
import com.cloud.app.system.utils.RedissonDelayQueueUtil;
import lombok.RequiredArgsConstructor;
import lombok.extern.slf4j.Slf4j;
import org.apache.commons.lang3.ObjectUtils;
import org.redisson.api.RBlockingDeque;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.boot.CommandLineRunner;
import org.springframework.stereotype.Component;

@Slf4j
@Component
@RequiredArgsConstructor
public class ResourceBackUpListener implements CommandLineRunner {

    private final RedissonDelayQueueUtil delayQueueUtil;

    public static final String RESOURCE_CANCEL_QUEUE ="cloud:resource:backup:queue";

    @Autowired
    private ICpInstanceInfoService cpInstanceInfoService;

    @Autowired
    private ICpAppInstanceBackInfoService cpAppInstanceBackInfoService;

    @Autowired
    private CpInstanceInfoMapper cpInstanceInfoMapper;

    @Autowired
    private CpAppInstanceBackInfoMapper cpAppInstanceBackInfoMapper;

    @Autowired
    private CpDiskInfoMapper cpDiskInfoMapper;


    @Override
    public void run(String... args) throws Exception {
        log.info("=== Redisson 延迟队列消费者启动 ===");

        // 获取就绪队列 (任务到期后会自动进入这里)
        // 泛型需要和生产者一致,如果不确定类型,可以用 String 接收 JSON
        RBlockingDeque<ResourceBackUpTask> readyQueue =
                delayQueueUtil.getReadyQueue(RESOURCE_CANCEL_QUEUE);

        while (!Thread.currentThread().isInterrupted()) {
            try {
                // 【核心】阻塞式获取任务
                // 如果队列为空,线程会挂起,不消耗 CPU,直到有新任务到期
                ResourceBackUpTask event = readyQueue.take();

                log.info("【收到到期任务】订单ID: {}, 用户: {}", event.getInstanceId(), event.getUserId());

                // 执行业务逻辑
                handleCancelTask(event);

            } catch (InterruptedException e) {
                Thread.currentThread().interrupt();
                log.warn("消费者线程被中断,停止运行");
                break;
            } catch (Exception e) {
                log.error("消费任务异常", e);
                // 发生异常时,任务已经出队。如果需要重试,需自行实现补偿机制或记录死信
            }
        }
        log.info("=== 消费者已停止 ===");
    }


    /**
     * 具体的业务处理逻辑
     */
    private void handleCancelTask(ResourceBackUpTask task) {
        try {
            // do something
           
            }
        } catch (Exception e) {
            log.error("任务失败: {}", task.getInstanceId(), e);
        }
    }
}
相关推荐
NoSi EFUL1 分钟前
MySQL中ON DUPLICATE KEY UPDATE的介绍与使用、批量更新、存在即更新不存在则插入
android·数据库·mysql
河阿里3 分钟前
SQL数据库:五大范式(NF)
数据库·sql·oracle
|晴 天|17 分钟前
Vue 3 + TypeScript + Element Plus 博客系统开发总结与思考
前端·vue.js·typescript
wuqingshun31415923 分钟前
说说mybatis的缓存机制
java·缓存·mybatis
空中海1 小时前
Kubernetes 生产实践、可观测性与扩展入门
java·贪心算法·kubernetes
猫3281 小时前
v-cloak
前端·javascript·vue.js
旷世奇才李先生1 小时前
Vue 3\+Vite\+Pinia实战:企业级前端项目架构设计
前端·javascript·vue.js
Devin~Y1 小时前
大厂Java面试实录:Spring Boot/Cloud、Kafka、Redis、K8s 与 Spring AI(RAG/Agent)三轮连环问
java·spring boot·redis·mysql·spring cloud·kafka·kubernetes
l1t1 小时前
DeepSeek总结的PostgreSQL 19查询提示功能
数据库·postgresql
bLEd RING1 小时前
SpringBoot3.3.0集成Knife4j4.5.0实战
java