1.使用场景?
消息队列延迟,比如订单模块支付超时...
2.延迟两种实现。
做消息延时有两种:死信队列 + 延迟插件。两种不同在于:死信队列是基于队列延时,延时插件是基于交换机延时。延时插件有一个小bug因为在延迟在交换机做消息没有到队列,所以RabbitTemplate.ReturnCallback会受到一个错误,需要自己进行处理。所以在项目中我选择了死信队列。
2.实现流程图。
上图解析:其实就是多了一个死信队列做延时等待,时间够了就重新把消息发送到交换机里面做消息派发。
上图所得我们需要创建几个东西:1.一个交换机(正常和死信同用一个)2.死信队列 3.正常交换机
3.直接上代码。
/** * @Auther chenXingHua * @Date 2021/1/22 * @fun 测试死信交换机 : 正常交换机-->>>bing 死信队列(延时等待) --->>>正常交换机---->>>正常队列 */ package com.wlkj.mq.configs; import org.springframework.amqp.core.*; import org.springframework.context.annotation.Bean; import org.springframework.context.annotation.Configuration; import java.util.HashMap; import java.util.Map; @Configuration public class TestDealConst { public static final String exchange_dead = "exchange.dead";//普通交换机 死信交换机 public static final String routing_dead_1 = "routing.dead.1";//死信路由 public static final String routing_dead_2 = "routing.dead.2"; public static final String queue_dead_1 = "queue.dead.1";//TTL public static final String queue_dead_2 = "queue.dead.2"; //死信队列 @Bean public Queue queue1() { Map<String, Object> arguments = new HashMap<>(); arguments.put("x-dead-letter-exchange", exchange_dead); //时间到了信息转到的交换机 arguments.put("x-dead-letter-routing-key", routing_dead_2); //路由key //全局设置过期时间,局部优先全局 arguments.put("x-message-ttl", 5 * 1000); return QueueBuilder.durable(queue_dead_1).withArguments(arguments).build(); } //普通队列 @Bean public Queue queue2() { return QueueBuilder.durable(queue_dead_2).build(); } //死信交换机/普通交换机 二合一 @Bean public Exchange exchange() { return ExchangeBuilder.directExchange(exchange_dead).build(); } //绑定死信队列 @Bean public Binding binding1(){ return BindingBuilder.bind(queue1()).to(exchange()).with(routing_dead_1).noargs(); } //绑定最后的队列 @Bean public Binding binding2(){ return BindingBuilder.bind(queue2()).to(exchange()).with(routing_dead_2).noargs(); } }