4.RabbitMQ 消息确认机制

4.RabbitMQ 消息确认机制

  • RabbitMQ在传递消息的过程中充当了代理人(Broker)的角色,那生产者(Producer)怎样知道消息被正确投递到 Broker了呢?
  • RabbitMQ提供了监听器(Listener)来接收消息投递的状态。
    消息确认涉及两种状态:Confirm与Return。

Confirm & Return

  • Confirm代表生产者将消息送到Broker时产生的状态,后续会出现两种情况:
    • ack 代表Broker已经将数据接收。
    • nack 代表Broker拒收消息。原因有多种,队列已满,限流,1O异常…
  • Return代表消息被Broker正常接收(ack)后,但Broker没有对应的队列进行投递时产生的状态,消息被退回给生产者。
  • 注意:上面两种状态只代表生产者与Broker之间消息投递的情况。与消费者是否接收/确认消息无关。

代码如下:

import com.rabbitmq.client.*;
import rabbitmq.ConnectionUtil;
import java.io.IOException;
import java.util.HashMap;
import java.util.Iterator;
import java.util.Map;

public class Confirm {
    public static void main(String[] args)  throws Exception{
        String exchangeStr = "exchange_weather_routing";
        Map<String,String> area = new HashMap();
        area.put("china.beijing.20221128","北京20221128号天气晴朗!");
        area.put("china.zhengzhou.20221128","郑州20221128号天气小雪!");
        area.put("us.NewYork.20221129","纽约20221129号天气晴朗!");
        area.put("us.Washington.20221129","华盛顿20221129号天气小雪!");

        Connection connection = ConnectionUtil.getConnection();
        Channel channel = connection.createChannel();
        //开启confirm 监听模式
        channel.confirmSelect();
        channel.addConfirmListener(new ConfirmListener() {
            public void handleAck(long l, boolean b) throws IOException {
                //第二个参数代表接收的数据是否为批量接收,一般用不到
                System.out.println("消息已经被接收,tag:"+l);
            }

            public void handleNack(long l, boolean b) throws IOException {
                System.out.println("消息已经被Broker拒收,tag:"+l);
            }
        });
        // 开启Return 监听
        channel.addReturnListener(new ReturnCallback() {
            public void handle(Return aReturn) {
                System.out.println("===================");
                System.out.println("Return编码:"+aReturn.getReplyCode()+"==Return 描述:"+aReturn.getReplyText());
                System.out.println("交换机:"+aReturn.getExchange()+"==路由Key:"+aReturn.getRoutingKey());
                System.out.println("Return主题:"+new String(aReturn.getBody()));
            }
        });
        Iterator<Map.Entry<String, String>> itr = area.entrySet().iterator();
        while (itr.hasNext()){
            Map.Entry<String, String> m = itr.next();
            //第一个参数:交换机名字,第二个参数:消息的Routing key
            channel.basicPublish(exchangeStr,m.getKey(),null,m.getValue().getBytes());
        }
        //不能关闭,关闭掉就监听不到了
/*        channel.close();
        connection.close();*/
    }
}

总结:消息确认的两种状态:Confirm与Return,是生产者的状态不是消费者的状态,消费者的状态为:basicAck方法。

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

苹水相峰

你的打赏是对我最大的肯定

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值