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);
        }
    }
}
相关推荐
h7ml1 小时前
企业微信API接口的数据一致性保障:Java Seata分布式事务在跨系统审批流程中的应用
java·分布式·企业微信
I_LPL1 小时前
day50 代码随想录算法训练营 图论专题3
java·算法·深度优先·图论·求职面试
维齐洛波奇特利(male)1 小时前
IDEA 实例类多开bug:勾选后还是只能运行一个类
java·bug·intellij-idea
蜜獾云1 小时前
设计模式之简单工厂模式(4):创建对象时不会暴露创建逻辑
java·设计模式·简单工厂模式
Luna-player1 小时前
npx create-vue 创建 Vue 3 项目的交互式配置界面
前端·javascript·vue.js
滴滴答滴答答1 小时前
机考刷题之 7 LeetCode 240 搜索二位矩阵Ⅱ
java·算法·leetcode
还是大剑师兰特1 小时前
const { proxy } = getCurrentInstance() 的正确使用方法
前端·javascript·vue.js
MX_93591 小时前
Spring的xml方式声明式事务控制
xml·java·后端·spring
进击的荆棘2 小时前
优选算法——模拟
java·开发语言·算法·模拟