当前位置:   article > 正文

Rabbit MQ------>延迟队列!!!

Rabbit MQ------>延迟队列!!!

一、场景:

1.定时发布文章

2.秒杀之后,给30分钟时间进行支付,如果30分钟后,没有支付,订单取消。

3.预约餐厅,提前半个小时发短信通知用户。

A -> 13:00 17:00 16:30 延迟时间: 7*30 * 60 * 1000

B -> 11:00 18:00 17:30 延迟时间: 13*30 * 60 * 1000

C -> 8:00 14:00 13:30 延迟时间: 11*30 * 60 * 1000

二、3种应用方式:

1、第一种方式:创建具有超时功能且绑定死信交换机的消息队列

  1. @Bean
  2. public Queue directQueueLong(){
  3. return QueueBuilder.durable("业务队列名称")
  4. .deadLetterExchange("死信交换机名称")
  5. .deadLetterRoutingKey("死信队列 RoutingKey")
  6. .ttl(20000) // 消息停留时间
  7. //.maxLength(500)
  8. .build();
  9. }

监听死信队列,即可处理超时的消息队列

缺点:

上述实现方式中,ttl延时队列中所有的消息超时时间都是一样的,如果不同消息想设置不一样的超时时间,就需要建立多个不同超时时间的消息队列,比较麻烦,且不利于维护。

2、第二种方式:创建通用延时消息

  1. rabbitTemplate.convertAndSend("交换机名称", "RoutingKey","对象",
  2. message => {
  3. message.getMessageProperties().setExpiration(String.valueOf(5000))
  4. return message;
  5. }
  6. );

缺点:

该种方式可以创建一个承载不同超时时间消息的消息队列,但是这种方式有一个问题,如果消息队列中排在前面的消息没有到超时时间,即使后面的消息到了超时时间,先到超时时间的消息也不会进入死信队列,而是先检查排在最前面的消息队列是否到了超时时间,如果到了超时时间才会继续检查后面的消息。

 3、第三种方式:使用rabbitmq的延时队列插件,实现同一个队列中有多个不同超时时间的消息,并按时间超时顺序出队

1)下载延迟插件:Releases · rabbitmq/rabbitmq-delayed-message-exchange · GitHub

 注意下载的插件版本要和你的MQ版本一致。

2)安装插件并启用

①、我用的是 Docker 客户端,下载完成后直接把插件放在 /root 目录,然后拷贝到容器内plugins目录下(rabbitmq是容器的name,也可以使用容器id)

docker cp /opt/rabbitmq/rabbitmq_delayed_message_exchange-3.10.0.ez rabbitmq:/plugins

②、进入 Docker 容器

docker exec -it rabbitmq /bin/bash

③、在plugins内启用插件

  1. #先执行,解除防火墙限制,增加文件权限
  2. umask 0022
  3. rabbitmq-plugins enable rabbitmq_delayed_message_exchange

④、退出容器

exit

⑤、重启容器

  1. docker restart 容器id/容器名
  2. docker restart rabbitmq

⑥、通过UI界面查看

⑦、原理解释

三、 java使用

1、消费者(需要定制一个交换机CustomExchange)

  1. /*
  2. * Copyright (c) 2020, 2024, All rights reserved.
  3. *
  4. */
  5. package com.by.consumer;
  6. import cn.hutool.core.map.MapUtil;
  7. import org.springframework.amqp.core.*;
  8. import org.springframework.amqp.rabbit.annotation.RabbitListener;
  9. import org.springframework.context.annotation.Bean;
  10. import org.springframework.context.annotation.Configuration;
  11. import java.util.Map;
  12. /**
  13. * <p>Project: spring-boot-rabbitMQ - DirectConsumer</p>
  14. * <p>Powered by scl On 2024-04-07 16:57:20</p>
  15. * <p>描述:<p>
  16. *
  17. * @author 孙臣龙 [1846080280@qq.com]
  18. * @version 1.0
  19. * @since 17
  20. */
  21. @Configuration
  22. public class DelayConsumer {
  23. //注册队列
  24. @Bean
  25. public Queue delayQueue() {
  26. return QueueBuilder.durable("Delay_Q01").build();
  27. }
  28. //注册交换机
  29. @Bean
  30. public CustomExchange delayExchange() {
  31. Map<String, Object> map = MapUtil.of("x-delayed-type", "direct");
  32. return new CustomExchange("Delay_E01", "x-delayed-message", true, false, map);
  33. }
  34. //绑定交换机和队列
  35. @Bean
  36. public Binding binding() {
  37. return BindingBuilder.bind(delayQueue()).to(delayExchange()).with("RK01").noargs();
  38. }
  39. //启动一个消费者
  40. @RabbitListener(queues = "Delay_Q01")
  41. public void receiveMessage(OrderKO msg) {
  42. System.out.println("消费者1:" + msg);
  43. }
  44. }

2、生产者

  1. /*
  2. * Copyright (c) 2020, 2024, All rights reserved.
  3. *
  4. */
  5. package com.by.provider;
  6. import com.by.consumer.OrderKO;
  7. import org.springframework.amqp.AmqpException;
  8. import org.springframework.amqp.core.Message;
  9. import org.springframework.amqp.core.MessagePostProcessor;
  10. import org.springframework.amqp.rabbit.core.RabbitTemplate;
  11. import org.springframework.beans.factory.annotation.Autowired;
  12. import org.springframework.stereotype.Service;
  13. /**
  14. * <p>Project: spring-boot-rabbitMQ - DirectProvider</p>
  15. * <p>Powered by scl On 2024-04-07 17:06:41</p>
  16. * <p>描述:<p>
  17. *
  18. * @author 孙臣龙 [1846080280@qq.com]
  19. * @version 1.0
  20. * @since 17
  21. */
  22. @Service
  23. public class DelayProvider {
  24. @Autowired
  25. private RabbitTemplate rabbitTemplate;
  26. //启动一个生产者
  27. public void send(OrderKO orderKO) {
  28. rabbitTemplate.convertAndSend("Delay_E01", "RK01", orderKO, new MessagePostProcessor() {
  29. @Override
  30. public Message postProcessMessage(Message message) throws AmqpException {
  31. int id = orderKO.getId();
  32. int delay = 0;
  33. if (id == 1) {
  34. delay = 5000;
  35. } else if (id==2) {
  36. delay = 4000;
  37. }else if (id==3) {
  38. delay = 3000;
  39. }else if (id==4) {
  40. delay = 2000;
  41. }else if (id==5) {
  42. delay = 1000;
  43. }
  44. //延迟交换机设置delay参数,设置消息的延迟时间,单位毫秒
  45. message.getMessageProperties().setDelay(delay);
  46. return message;
  47. }
  48. });
  49. }
  50. }

3、测试

  1. /**
  2. * 延迟交换机
  3. * @throws InterruptedException
  4. * @throws IOException
  5. */
  6. @Test
  7. void test5() throws InterruptedException, IOException {
  8. for (int i = 1; i <= 5; i++) {
  9. OrderKO orderKO = OrderKO.builder().id(i).name("孙臣龙" + i).build();
  10. System.out.println("发送消息"+i);
  11. delayProvider.send(orderKO);
  12. }
  13. Thread.sleep(10000);
  14. //System.in.read();
  15. }

4、结果展示

声明:本文内容由网友自发贡献,不代表【wpsshop博客】立场,版权归原作者所有,本站不承担相应法律责任。如您发现有侵权的内容,请联系我们。转载请注明出处:https://www.wpsshop.cn/w/知新_RL/article/detail/438395
推荐阅读
相关标签
  

闽ICP备14008679号