RabbitMQ报错:Shutdown Signal channel error; protocol method

报错信息:

Shutdown Signal: channel error; protocol method: #method<channel.close>(reply-code=406, reply-text=PRECONDITION_FAILED - unknown delivery tag 1, class-id=60, method-id=80)

原因

默认情况下 RabbitMQ 是自动ACK(确认签收)机制,就意味着 MQ 会在消息发送完毕后,自动帮我们去ACK(确认),若是在代码中再手动确认签收,就会造成确认错误。

"PRECONDITION_FAILED - unknown delivery tag" 表明交付标签(delivery tag)不合法或已经被确认过

因此我们需要在订阅者(消费者)的方法上标识,消息手动确认签收ackMode = "MANUAL",代码如下:

复制代码
    @RabbitListener(queues = RabbitMqContants.DIRECT_QUEUE1_NAME,ackMode = "MANUAL")
    public void directConsumerA(String msg, Channel channel, Message message) throws IOException {

        try {
            // 处理消息
            log.info("=================DIRECT-队列1-消费者A 接收到消息:msg = {}", msg);
            //消息消费手动确认
            channel.basicAck(message.getMessageProperties().getDeliveryTag(), false);
        } catch (Exception e) {
            e.printStackTrace();
            // 处理异常,可能需要重新将消息放回队列
            channel.basicNack(message.getMessageProperties().getDeliveryTag(), false, true);
        }
    }

关于rabbitMQ在yml配置手动ack不生效,重复消费的问题

复制代码
spring:
 rabbitmq:
    host: xxx
    port: 5672
    username: guest
    password: guest
    listener:
      direct:
        acknowledge-mode: manual # 手动ACK开启:自动确认 none ;手动确认 manual ;根据异常情况确认 auto (较麻烦,不推荐)
此处已经设置了手动答应,消费者代码是这样的:
复制代码
@RabbitListener(queues = RabbitMqContants.DIRECT_QUEUE1_NAME)
    public void directConsumerA(String msg, Channel channel, Message message) throws IOException {

        try {
            // 处理消息
            log.info("=================DIRECT-队列1-消费者A 接收到消息:msg = {}", msg);
            //消息消费手动确认
            channel.basicAck(message.getMessageProperties().getDeliveryTag(), false);
        } catch (Exception e) {
            e.printStackTrace();
            // 处理异常,可能需要重新将消息放回队列
            channel.basicNack(message.getMessageProperties().getDeliveryTag(), false, true);
        }
    }
然而消费者会重复消费消息,控制台还是会打印如下异常:
复制代码
Channel shutdown: channel error; protocol method: #method<channel.close>(reply-code=406, reply-text=PRECONDITION_FAILED - unknown delivery tag 1, class-id=60, method-id=80)

查找资料和测试发现,rabbitmq默认是用的SimpleRabbitListenerContainerFactory

配置文件中如果不指定监听容器类型的话,配置是不会生效的,

只要配置文件配置改成以下配置,这个问题就可以解决了。

复制代码
spring:
 rabbitmq:
    host: xxx
    port: 5672
    username: guest
    password: guest
    listener:
      # 设置监听容器(Listener container)类型,如不设置,将会默认为SimpleRabbitListenerContainerFactory,且下面的direct配置不生效
      type: direct
      direct:
        acknowledge-mode: manual # 手动ACK开启:自动确认 none ;手动确认 manual ;根据异常情况确认 auto (较麻烦,不推荐)
重点是type这个配置,如果不设置type的话,要将direct改成simple才会生效

总结

详情请参考:
RabbitMQ报错:Shutdown Signal: channel error; protocol method
关于rabbitMQ在yml配置手动ack不生效,重复答应的问题

相关推荐
可观测性用观测云7 小时前
阿里云 RabbitMQ 可观测性最佳实践
rabbitmq
搞数据的小杰8 小时前
spark广播表大小超过Spark默认的8GB限制
大数据·数据库·分布式·spark
isNotNullX8 小时前
数据怎么分层?从ODS、DW、ADS三大层一一拆解!
大数据·开发语言·数据仓库·分布式·spark
Akamai中国9 小时前
为何说分布式 AI 推理已成为下一代计算方式
人工智能·分布式·云计算·边缘计算·云服务·akamai
腾讯云中间件9 小时前
深度剖析 TDMQ RabbitMQ 版经典队列底层存储机制
消息队列·rabbitmq·腾讯
小马爱打代码12 小时前
分布式通信框架 - JGroups
分布式·节点通信
Bug退退退12313 小时前
RabbitMQ 之顺序性保障
服务器·分布式·rabbitmq
羌俊恩16 小时前
分布式存储之Ceph使用指南--部署篇(未完待续)
分布式·ceph·pg·osd·rados
wb18916 小时前
LVS的集群技术和分布式
运维·笔记·分布式·云计算·lvs