消息确认的几点说明:

1. Basic.Ack 发回给 RabbitMQ 以告知,可以将相应 message 从 RabbitMQ 的消息缓存中移除。
2. Basic.Ack 未被 consumer 发回给 RabbitMQ 前出现了异常,RabbitMQ 发现与该 consumer 对应的连接被断开,之后将该  message 以轮询方式发送给其他 consumer (假设存在多个 consumer 订阅同一个 queue)。
3. 在 no_ack=true 的情况下,RabbitMQ 认为 message 一旦被 deliver 出去了,就已被确认了,所以会立即将缓存中的 message 删除。所以在 consumer 异常时会导致消息丢失。
4. 来自 consumer 侧的 Basic.Ack 与 发送给 Producer 侧的 Basic.Ack 没有直接关系。


根据消费方不同的确认模式,确认时机也不同.

自动确认会在消息发送给消费者后立即确认,如果手动则当消费者调用ack,nack,reject几种方法时进行确认.


一般会设置手动模式,业务失败后可以进行一些操作.

1. 在这里不提如何集成rabbit mq到spring

2. 实现功能的配置都在消费者端:

3. 下面是步骤和说明

(1)在消费者端的mq配置文件上添加,配置  关键代码为 acknowledeg = "manual"

,意为表示该消费者的ack方式为手动(此时的queue已经和生产者的exchange通过某个routeKey绑定了)

消费者的确认模式需要配置为manual,其中确认模式包括NONE,MANUL,与AUTO三种


[html]  view plain  copy


 

1. <rabbit:listener-container connection-factory="connectionFactory" acknowledge="manual">  
2. <rabbit:listener queues="queue_xxx" ref="MqConsumer"/>  
3. <rabbit:listener queues="queue_xxx" ref="MqConsumer2"/>  
4. </rabbit:listener-container>


(2)新建一个类 MqConsumer ,并实现接口  ChannelAwareMessageListener ,实现onMessage方法,不需要指定方法。

因为下方图所示,springAMQP中已经实现了一个功能,如果该监听器已经实现了下面2个接口,则直接调用onMessage方法

Java rabbitmq消费者监听原理 rabbitmq 消费者确认_spring

(3)关键点在实现了ChannelAwareMessageListener的onMessage方法后,会有2个参数。

一个是message(消息实体),一个是channel就是当前的通道

很多地方都没有说清楚怎么去手动ack,其实手动ack就是在当前channel里面调用basicAsk的方法,并传入当前消息的tagId就可以了。

Java rabbitmq消费者监听原理 rabbitmq 消费者确认_spring_02

basicAck()方法解析:





Java rabbitmq消费者监听原理 rabbitmq 消费者确认_缓存_03




第一个参数 deliveryTag:就是接受的消息的deliveryTag,可以通过msg.getMessageProperties().getDeliveryTag()获得


     第二个参数 multiple:如果为true,确认之前接受到的消息;如果为false,只确认当前消息。如果为true就表示连续取得多条消息才发会确认,和计算机网络的中tcp协议接受分组的累积确认十分相似,能够提高效率


同样的,如果要nack或者拒绝消息(reject)的时候,也是调用channel里面的basicXXX方法就可以了(要指定tagId)。

注意如果抛异常或nack(并且requeue为true),消息会重新入队列,并且会造成消费者不断从队列中读取同一条消息的假象。

Java rabbitmq消费者监听原理 rabbitmq 消费者确认_配置文件_04


//消息的标识,false只确认当前一个消息收到,true确认所有consumer获得的消息
channel.basicAck(message.getMessageProperties().getDeliveryTag(), false);
//ack返回false,并重新回到队列,api里面解释得很清楚
channel.basicNack(message.getMessageProperties().getDeliveryTag(), false, true);
//拒绝消息
channel.basicReject(message.getMessageProperties().getDeliveryTag(), true);