• rabbitMq集成Spring后,消费者设置手动ack,并且在业务上控制是否ack


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

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

    3. 下面是步骤和说明

    (1)在消费者端的mq配置文件上添加,配置  关键代码为 acknowledeg = "manual"
    ,意为表示该消费者的ack方式为手动(此时的queue已经和生产者的exchange通过某个routeKey绑定了)

    [html] view plain copy
     在CODE上查看代码片派生到我的代码片
    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方法


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

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

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


    basicAck()方法解析:



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

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


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

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



    //消息的标识,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);

  • 相关阅读:
    FFmpeg入门,简单播放器
    Linux系统编译Win32版本adb
    检测目标程序ELF bit是32还是64
    Swift编程资料全集
    Swift编程资料总结
    cocos2d-html5学习之三-为sprite添加触摸事件
    Cocos2d-html5学习笔记二
    cocos2d-x学习笔记一
    NSViewAnimation进行视图和窗口动画
    Cocoa中NSAnimation动画简介
  • 原文地址:https://www.cnblogs.com/chenny3/p/10226169.html
Copyright © 2020-2023  润新知