RabbitMQ如何实现延迟任务?

开发 前端
实现 RabbitMQ 延迟队列目前主流的实现方式,是采用官方提供的延迟插件来实现。而延迟插件需要先下载插件、然后配置并重启 RabbitMQ 服务,之后就可以通过编写代码的方式实现延迟队列了。

延迟任务是指当消息被发送以后,并不是立即执行,而是等待特定的时间后,消费者才会执行该消息。延迟队列的使用场景有以下几种:

  • 未按时支付的订单,30 分钟过期之后取消订单。
  • 给活跃度比较低的用户间隔 N 天之后推送消息,提高活跃度。
  • 新注册会员的用户,等待几分钟之后发送欢迎邮件等。

1.如何实现延迟队列?

延迟队列有以下两种实现方式:

  • 通过消息过期后进入死信交换器,再由交换器转发到延迟消费队列,实现延迟功能;
  • 使用官方提供的延迟插件实现延迟功能。

早期,大部分公司都会采用第一种方式,但因为第一种方式存在“队头阻塞”问题,所以在生产环境我们通常会使用 RabbitMQ 3.5.7(2015 年底发布)的延迟插件来实现,它的使用更简单、更方便,所以本文主要讲第二种实习方式。

2.实现延迟队列

2.1 安装并启动延迟队列

2.1.1 下载延迟插件

https://github.com/rabbitmq/rabbitmq-delayed-message-exchange/releases

注意:需要根据你自己的 RabbitMQ 服务器端版本选择相同版本的延迟插件,可以在 RabbitMQ 控制台查看:

图片图片

图片图片

2.1.2 将插件放到插件目录

接下来,将上一步下载的插件放到 RabbitMQ 服务器安装目录,如果是 docker,使用一下命令复制:

docker cp 宿主机文件 容器名称或ID:容器目录

如下图所示:

图片图片

之后,进入 docker 容器,查看插件中是否包含延迟队列:

docker exec -it 容器名称或ID /bin/bash rabbitmq-plugins list

如下图所示:

图片图片

2.1.3 启动插件

rabbitmq-plugins enable rabbitmq_delayed_message_exchange

如下图所示:

图片图片

2.1.4 重启RabbitMQ服务

安装完 RabbitMQ 插件之后,需要重启 RabbitMQ 服务才能生效。如果使用的是 Docker,只需要重启 Docker 容器即可:

docker restart 容器名称或ID

如下图所示:

图片图片

2.1.5 验收结果

在 RabbitMQ 控制台查看,新建交换机时是否有延迟消息选项,如果有就说明延迟消息插件已经正常运行了,如下图所示:

图片图片

2.1.6 手动创建延迟交换器(可选)

此步骤可选(非必须),因为某些版本下通过程序创建延迟交换器可能会出错,如果出错了,手动创建延迟队列即可,如下图所示:

图片图片

2.2 编写延迟消息实现代码

2.2.1 配置交换器和队列

import org.springframework.context.annotation.Configuration;
import org.springframework.amqp.core.*;
import org.springframework.context.annotation.Bean;

/**
 * 延迟交换器和队列
 */
@Configuration
public class DelayedExchangeConfig {
    public static final String EXCHANGE_NAME = "myDelayedExchange";
    public static final String QUEUE_NAME = "delayed.queue";
    public static final String ROUTING_KEY = "delayed.routing.key";

    @Bean
    public CustomExchange delayedExchange() {
        return new CustomExchange(EXCHANGE_NAME,
                "x-delayed-message", // 消息类型
                true, // 是否持久化
                false); // 是否自动删除
    }

    @Bean
    public Queue delayedQueue() {
        return QueueBuilder.durable(QUEUE_NAME)
                .withArgument("x-delayed-type", "direct")
                .build();
    }

    @Bean
    public Binding delayedBinding(Queue delayedQueue,CustomExchange delayedExchange) {
        return BindingBuilder.bind(delayedQueue()).to(delayedExchange()).with(ROUTING_KEY).noargs();
    }
}

2.1.2 定义消息发送方法

import org.springframework.amqp.rabbit.core.RabbitTemplate;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.scheduling.annotation.Scheduled;
import org.springframework.stereotype.Component;

@Component
public class DelayedMessageProducer {

    @Autowired
    private RabbitTemplate rabbitTemplate;

    @Scheduled(fixedDelay = 5000)
    public void sendDelayedMessage(String message) {
        rabbitTemplate.convertAndSend(DelayedExchangeConfig.EXCHANGE_NAME,
                DelayedExchangeConfig.ROUTING_KEY,
                message,
                messagePostProcessor -> {
                    messagePostProcessor.getMessageProperties().setDelay(10000); // 设置延迟时间,单位毫秒
                    return messagePostProcessor;
                });
    }
}

2.1.3 发送延迟消息

import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.web.bind.annotation.GetMapping;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.RequestParam;
import org.springframework.web.bind.annotation.RestController;

@RestController
@RequestMapping("/delayed")
public class DelayedMessageController {
    @Autowired
    private DelayedMessageProducer delayedMessageProducer;

    @GetMapping("/send")
    public String sendDirectMessage(@RequestParam String message) {
        delayedMessageProducer.sendDelayedMessage(message);
        return "Delayed message sent to Exchange: " + message;
    }
}

2.1.4 接收延迟消息

import org.springframework.amqp.rabbit.annotation.RabbitListener;
import org.springframework.stereotype.Component;


@Component
public class DelayedMessageConsumer {

    @RabbitListener(queues = DelayedExchangeConfig.QUEUE_NAME)
    public void receiveDelayedMessage(String message) {
        System.out.println("Received delayed message: " + message);
    }
}

PS:防止失联,请加我:vipStone

小结

实现 RabbitMQ 延迟队列目前主流的实现方式,是采用官方提供的延迟插件来实现。而延迟插件需要先下载插件、然后配置并重启 RabbitMQ 服务,之后就可以通过编写代码的方式实现延迟队列了。


责任编辑:武晓燕 来源: 磊哥和Java
相关推荐

2023-09-05 15:48:14

RabbitMQ延迟队列

2024-12-17 15:39:33

2019-02-25 15:44:16

开源RabbitMQSpring Clou

2024-01-26 13:16:00

RabbitMQ延迟队列docker

2024-04-28 08:52:33

RabbitMQ延迟队列延迟插件

2024-10-22 16:39:07

2024-04-09 10:40:04

2024-04-19 00:47:07

RabbitMQ消息机制

2021-12-08 10:47:35

RabbitMQ 实现延迟

2023-10-23 10:02:58

RabbitMQ延迟队列

2023-10-10 13:39:53

Spring队列优化

2024-10-16 09:29:30

RabbitMQ延迟队列

2024-11-05 16:58:21

RabbitMQ订单超时取消延迟队列

2021-10-15 10:39:43

RabbitMQ队列延迟

2022-05-31 09:36:18

JDKDelayQueueRedis

2023-02-27 22:03:06

数据库内存RocketMQ

2022-08-02 11:27:25

RabbitMQ消息路由

2022-06-27 23:49:21

数据仓库资源不足集群

2023-08-08 08:28:03

消息消费端Spring

2021-05-12 22:07:43

并发编排任务
点赞
收藏

51CTO技术栈公众号