rabbitmq死信交换机,死信队列使用

背景

对于核心业务需要保证消息必须正常消费,就必须考虑消费失败的场景,rabbitmq提供了以下三种消费失败处理机制

  1. 直接reject,丢弃消息(默认)
  2. 返回nack,消息重新入队列
  3. 将失败消息投递到指定的交换机
    对于核心业务,第一种方法显然不可接受,第二种方法如果代码有异常导致消费一直失败就会出现不断失败重新入队列的死循环问题,较好的方案是3,待消费失败问题修复后将消息从死信队列取出发回原队列重新消费。

实现

  • rabbit版本
xml 复制代码
<dependency>
  <groupId>org.springframework.boot</groupId>
  <artifactId>spring-boot-starter-amqp</artifactId>
  <version>2.6.3</version>
</dependency>
  • 配置死信交换机,路由,队列

  • 配置延迟消息业务队列消费失败投递到死信队列
java 复制代码
@Bean("orderCloseQueue")
    public Queue orderCloseQueue() {
        return QueueBuilder.durable(OrderRabbitConstants.ORDER_CLOSE_QUEUE)
            .deadLetterExchange(RabbitMqConstants.DEAD_LETTER_EXCHANGE)
            .deadLetterRoutingKey(RabbitMqConstants.DEAD_LETTER_ROUTING_KEY)
            .build();
    }
  • 配置手动返回ACK
java 复制代码
@Bean(name = {"manualContainerFactory"})
public SimpleRabbitListenerContainerFactory manualContainerFactory(@Qualifier("connectionFactory") ConnectionFactory connectionFactory) {
	SimpleRabbitListenerContainerFactory factory = new SimpleRabbitListenerContainerFactory();
	this.manualFactoryConfigurer.configure(factory, connectionFactory);
	factory.setAcknowledgeMode(AcknowledgeMode.MANUAL);
	factory.setDefaultRequeueRejected(this.enableRequeueRejected);
	if (this.enableConsumers) {
		factory.setConcurrentConsumers(this.concurrentConsumers);
		factory.setMaxConcurrentConsumers(this.maxConcurrentConsumers);
		factory.setPrefetchCount(this.prefetchCount);
	}
	return factory;
}
  • 业务队列消息消费模拟失败
java 复制代码
@RabbitListener(queues = OrderRabbitConstants.ORDER_CLOSE_QUEUE, containerFactory = "manualContainerFactory")
    public void consumerCloseOrder(Message message, Channel channel) throws IOException {
        String orderCode = new String(message.getBody(), CharsetUtil.UTF_8);
        String messageId = message.getMessageProperties().getMessageId();
        log.info("收到MQ messageId[{}],订单号[{}]", messageId, orderCode);
        if (true) {
            channel.basicReject(message.getMessageProperties().getDeliveryTag(), false);
            return;
        }
    }
  • 效果



    可以看到死信队列dead.letter.queue已经正常收到死信消息
  • 编写逻辑将死信消费推回原队列
java 复制代码
for (int i = 0; i < 10_000; i++) {
     Message message = rabbitTemplate.receive(RabbitMqConstants.DEAD_LETTER_QUEUE);
     if (message == null) {
         return String.format("完成%d条", i);
     }
     log.info("拉取死信消息:[{}]", message);
     try {
         Map<String, Object> headers = message.getMessageProperties().getHeaders();
         Map<String, Object> deathMap = ((List<Map<String, Object>>) headers.get("x-death")).get(0);
         String exchange = deathMap.get("exchange").toString();
         String routingKey = ((List) deathMap.get("routing-keys")).get(0).toString();
         rabbitTemplate.send(exchange, routingKey, message);
     } catch (Exception ex) {
         log.error("消费死信消息失败", ex);
         rabbitTemplate.send(RabbitMqConstants.DEAD_LETTER_EXCHANGE, RabbitMqConstants.DEAD_LETTER_ROUTING_KEY, message);
         return "重入队列异常";
     }
 }
  • 重推回业务队列效果

相关推荐
JLWcai2025100913 天前
铸造领域树脂砂轮|金利威多场景解决方案,20 + 配方覆盖全需求
mongodb·zookeeper·eureka·spark·rabbitmq·memcached·storm
风吹夏回14 天前
RabbitMQ 核心术语 + Python pika 方法完整讲解
分布式·python·rabbitmq
风吹夏回14 天前
RabbitMQ 三种模式入门:HelloWorld、WorkQueue、PubSub
分布式·rabbitmq·ruby
霸道流氓气质14 天前
分布式追踪与 RequestId 传播完全指南
分布式
cheems952714 天前
[RabbitMQ高级特性] 消息确认机制:从 Ready / Unacked 到 basicAck、basicReject、basicNack 的底层拆解
分布式·rabbitmq·ruby
枫华落尽14 天前
【Hadoop01-完全分布式运行模式】
分布式
隔壁阿布都14 天前
ShedLock 分布式定时任务锁框架介绍
spring boot·分布式
文艺倾年14 天前
【强化学习】数学推导专题,20W字总结(十五)
人工智能·分布式·大模型·强化学习·vibecoding
ACP广源盛1392462567314 天前
GSV9001S@ACP#1080P 级视频处理芯片,物理 AI 普及终端的高性价比选择
大数据·人工智能·分布式·嵌入式硬件·spark
guslegend14 天前
第1章:初始Kafka
分布式·kafka