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 "重入队列异常";
     }
 }
  • 重推回业务队列效果

相关推荐
g***B7386 小时前
后端在分布式中的服务配置
分布式
n***i957 小时前
后端在分布式缓存中的一致性哈希
分布式·缓存·哈希算法
O***p6047 小时前
后端在分布式中的服务治理
分布式
F***c32512 小时前
PHP在微服务中的分布式跟踪
分布式·微服务·php
2501_9411458515 小时前
Go语言高效爬虫开发实战:协程与并发请求代码解析
rabbitmq
深蓝电商API15 小时前
Scrapy + Scrapy-Redis 分布式爬虫集群部署(2025 最新版)
redis·分布式·scrapy
Sinowintop16 小时前
易连EDI-EasyLink无缝集成之消息队列Kafka
分布式·网络协议·kafka·集成·国产化·as2·国产edi
玩转以太网16 小时前
W55MH32 单芯片以太网方案:破解分布式 IO 三大痛点
分布式·物联网
小坏讲微服务18 小时前
Spring Cloud Alibaba 整合 Scala 教程完整使用
java·开发语言·分布式·spring cloud·sentinel·scala·后端开发
pale_moonlight18 小时前
九、Spark基础环境实战((上)虚拟机安装Scala与windows端安装Scala)
大数据·分布式·spark