springBoot-rabbit MQ-设置手动确认ACK-Channel shutdown异常

记一次 springBoot + rabbitMQ设置手动ACK时的异常,配置文件设置了

spring.rabbitmq.listener.simple.acknowledge-mode=manual//表示启动消费方的手动确认

但手动ACK没有生效,代码接收消息后,当业务完毕,执行:

channel.basicAck(message.getMessageProperties().getDeliveryTag(), false);//确认消息已经被消费

异常为:

2018-11-05 17:33:33.090 [ERROR] [AMQP Connection 59.110.229.19:5672] Caller+0     at org.springframework.amqp.rabbit.connection.CachingConnectionFactory$DefaultChannelCloseLogger.log(CachingConnectionFactory.java:1199)
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)
2018-11-05 17:33:34.081 [WARN ] [SimpleAsyncTaskExecutor-2] Caller+0     at org.springframework.amqp.rabbit.listener.SimpleMessageListenerContainer$AsyncMessageProcessingConsumer.logConsumerException(SimpleMessageListenerContainer.java:1462)
Consumer raised exception, processing can restart if the connection factory supports it
com.rabbitmq.client.ShutdownSignalException: channel error; protocol method: #method<channel.close>(reply-code=406, reply-text=PRECONDITION_FAILED - unknown delivery tag 1, class-id=60, method-id=80)
    at com.rabbitmq.client.impl.ChannelN.asyncShutdown(ChannelN.java:484)
    at com.rabbitmq.client.impl.ChannelN.processAsync(ChannelN.java:321)
    at com.rabbitmq.client.impl.AMQChannel.handleCompleteInboundCommand(AMQChannel.java:144)
    at com.rabbitmq.client.impl.AMQChannel.handleFrame(AMQChannel.java:91)
    at com.rabbitmq.client.impl.AMQConnection$MainLoop.run(AMQConnection.java:560)
    at java.lang.Thread.run(Thread.java:748)
2018-11-05 17:33:34.082 [INFO ] [SimpleAsyncTaskExecutor-2] Caller+0     at org.springframework.amqp.rabbit.listener.SimpleMessageListenerContainer$AsyncMessageProcessingConsumer.run(SimpleMessageListenerContainer.java:1453)
Restarting Consumer: tags=[{amq.ctag-zms15dchTPRiv8SH2W0_8Q=videoChange}], channel=Cached Rabbit Channel: AMQChannel(amqp://admin@59.110.229.19:5672/,2), conn: Proxy@131c5bd Shared Rabbit Connection: SimpleConnection@3a500321 [delegate=amqp://admin@59.110.229.19:5672/, localPort= 50032], acknowledgeMode=AUTO local queue size=0
 

体现重点的有两条:

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

表明这条消息对于MQ来说没找到,重点是这个---unknown delivery tag 1-参考rabbitMQ的tag机制可知这条消息已经完成了消费。

2、Restarting Consumer: tags=[{amq.ctag-zms15dchTPRiv8SH2W0_8Q=videoChange}], channel=Cached Rabbit Channel: AMQChannel(amqp://admin@59.110.229.19:5672/,2), conn: Proxy@131c5bd Shared Rabbit Connection: SimpleConnection@3a500321 [delegate=amqp://admin@59.110.229.19:5672/, localPort= 50032], acknowledgeMode=AUTO local queue size=0

表明这条消息被一个设置为自动确认的任务(至于他具体是啥,怎么执行的,没深入看源码,等大神给解释了。)给确认了。

各种百度,发现MQ配置的时候,如果配置了json解析器。如下:

@Bean
public MessageConverter messageConverter() {
    return new ContentTypeDelegatingMessageConverter(new Jackson2JsonMessageConverter());
}

则程序会走自动确认消费,配置文件的配置就不生效了(这个不知道为什么,也可能springboot中某些配置的先后顺序的问题)

这个问题解决办法就是重写一下这个东东,用代码设置手动确认。就OK了

@Bean
public RabbitListenerContainerFactory<?> rabbitListenerContainerFactory(ConnectionFactory connectionFactory){
    SimpleRabbitListenerContainerFactory factory = new SimpleRabbitListenerContainerFactory();
    factory.setConnectionFactory(connectionFactory);
    factory.setMessageConverter(new Jackson2JsonMessageConverter());
    factory.setAcknowledgeMode(AcknowledgeMode.MANUAL);             
    return factory;
}

 

  • 21
    点赞
  • 26
    收藏
    觉得还不错? 一键收藏
  • 11
    评论
在使用Spring Boot集成RabbitMQ时,可以使用手动确认模式来确保消息被正确处理。手动确认模式通过调用channel.basicAck方法来确认消息的处理结果。如果手动确认无效,可能是由于以下几个原因: 1. 消费者没有配置正确的ack模式:在RabbitMQ配置中,可以使用在@RabbitListener注解中设置ackMode属性来指定确认模式。常见的确认模式有NONE、AUTO、MANUAL三种。确保消费者的确认模式设置为MANUAL。 2. 消费者没有正确地处理消息:在消费者代码中,确保消息的处理结果正确返回,并在处理完成后调用channel.basicAck方法进行手动确认。如果消息处理抛出异常或者没有调用basicAck方法,消息会被认为是未处理完成,RabbitMQ会将消息重新分发给其他消费者。 3. 消息被重新入队列:如果消息被重新入队列,可能会导致手动确认无效。在处理消息过程中,如果消费者抛出了异常或者手动拒绝了消息,RabbitMQ会将消息重新放回队列。确保在处理消息过程中没有发生异常,并且手动拒绝消息时设置requeue参数为false,确保消息不会重新入队列。 4. RabbitMQ服务配置问题:检查RabbitMQ服务器的配置,确保可靠性模式设为ACK。如果服务端配置的不正确,可能导致消费者的手动确认无效。 总之,如果手动确认无效,需要仔细检查消费者的确认模式、消息处理过程中的异常处理、消息的手动拒绝和确认操作等方面,同时确保RabbitMQ服务器的配置正确。
评论 11
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值