-
关于rabbitMQ在yml配置手动ack不生效,重复答应的问题
- 在yml文件中配置了手动答应,但是消费者消费完成时,却还是自动答应
- 以下是yml配置
-
#rabbit配置 spring: rabbitmq: # rabbit地址 host: 127.0.0.1 # rabbit端口号 port: 5672 # 用户账号和密码 username: guest password: guest #rabbit项目名,每个virtualHost的队列是隔离的,相当于数据库 virtual-host: /rabbit #开启Publisher Confirms 模式,消息发送到交换器后触发回调。 #publisher-confirms: true #开启PublisherReturn 模式,交换机将消息发送到对应队列失败时触发 #publisher-returns: true listener: direct: #设置监听为手动答应模式 acknowledge-mode: manual - 此处已经设置了手动答应,消费者代码是这样的:
-
/** * 推送队列的监听 * 注解中为监听的队列 * @param * @param channel * @param tag * @throws IOException */ @RabbitListener(queues = "rabbit.direct.queue.push") public void processDirect(String str, Channel channel, @Header(AmqpHeaders.DELIVERY_TAG) long tag) throws IOException { System.out.println("收到消息"); System.out.println(str); //手动答应消费完成,从队列中删除该消息 channel.basicAck(tag,true); } - 源码看不太懂,但是打断点发现,在下图中,basicAck在手动答应时[channel.basicAck(tag,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
- 配置文件中如果不指定监听容器类型的话,配置是不会生效的,
- 只要配置文件配置改成以下配置,这个问题就可以解决了
-
#rabbit配置 spring: rabbitmq: # rabbit地址 host: 127.0.0.1 # rabbit端口号 port: 5672 # 用户账号和密码 username: guest password: guest #rabbit项目名,每个virtualHost的队列是隔离的,相当于数据库 virtual-host: /rabbit #开启Publisher Confirms 模式,消息发送到交换器后触发回调。 #publisher-confirms: true #开启PublisherReturn 模式,交换机将消息发送到对应队列失败时触发 #publisher-returns: true listener: #设置监听容器(Listener container)类型,如不设置,将会默认为SimpleRabbitListenerContainerFactory,且下面的direct配置不生效 type: direct direct: #设置监听为手动答应模式 acknowledge-mode: manual - 重点是type这个配置,如果不设置type的话,要将direct改成simple才会生效
来源:oschina
链接:https://my.oschina.net/u/4253180/blog/4791223
