引言

RabbitMQ的模型是生产者发送信息到 Broker (代理),消费者从 Broker 中取出信息。但是生产者怎么知道消息是否真的发送到 Broker 中了呢?Broker 又怎么知道消息到底有没有被消费者消费?

如果由于网络原因出现故障,生产者生产的消息未到达 Broker 或者 Broker 的消息被虚假消费,而它们又不知道,就会产生很严重的问题,如重复消费等。

RabbitMQ的消息确认流程

从图中可以看出:

消息确认机制分为生产者确认和消费者确认

  • ConfirmCallback 生产者
  • ReturnCallback 生产者
  • ACK 消费者

生产者确认

  • 消息到达RabbitMQ的Exchange:Exchange向生产者发送Confirm确认。成功抑或失败都会返回一个confirmCallback
  • 消息成功达到Exchange,但是从Exchange投递Queue失败:向生产者返回一个returnCallback。只有失败才会返回

消费者确认

  • 消费者收到消息后需要对 RabbitMQ Server 进行消息 ACK 确认,RabbitMQ 根据确认信息决定是删除队列中的该信息还是重新发送

代码实现

生产者确认

重点在于生产者重写下面两个方法

  • rabbitMQTemplate.setConfirmCallback

  • rabbitMQTemplate.setReturnCallback

  1. 开启生产者消息确认

    spring:
    rabbitmq:
    host: localhost
    port: 5672
    virtual-host: /
    username: root
    password: root
    # 开启两个模式的生产者消息确认
    publisher-confirm-type: simple
    publisher-returns: true
  2. 声明交换机、队列,绑定交换机和队列

    @Configuration
    public class RabbitMQConfig { private static final String SB_TOPIC_EXCHANGE="sb_topic_exchange";
    private static final String SB_TOPIC_QUEUE="sb_topic_queue1"; // 注入交换机 topic类型
    @Bean("topicExchange")
    public Exchange topicExchange(){
    return ExchangeBuilder.topicExchange(SB_TOPIC_EXCHANGE).durable(true)
    .autoDelete().build();
    }
    // 声明队列
    @Bean
    public Queue queue1(){
    return QueueBuilder.durable(SB_TOPIC_QUEUE).build();
    } // 绑定队列和交换机
    @Bean
    public Binding exchangQueue(@Qualifier("queue1") Queue queue, @Qualifier("topicExchange") Exchange exchange){
    return BindingBuilder.bind(queue).to(exchange).with("user.#").noargs();
    }
    }
  3. 创建消费者

    @Component
    @RabbitListener(queues = "sb_topic_queue1")
    public class Consumer { @RabbitHandler
    public void testPublishConfirm(String msg) {
    System.out.println("收到的信息:"+msg);
    }
    }
  4. 创建生产者

    创建生产者发送消息到消息队列,模拟两种异常情况

    @SpringBootTest
    class RabiitmqSpringbootApplicationTests { @Autowired
    RabbitTemplate template; @Test
    void testConfirmTrue() {
    // 设置confirm回调函数
    template.setConfirmCallback(new RabbitTemplate.ConfirmCallback() {
    @Override
    public void confirm(CorrelationData correlationData, boolean b, java.lang.String s) {
    if (b) System.out.println("消息发送成功");
    else System.out.println("消息发送失败");
    } });
    // 模拟生产者发送信息--正常情况
    template.convertAndSend("sb_topic_exchange","user.info","日志级别:info;日志模块:user;日志信息:*****");
    } @Test
    void testConfirmFalse() {
    // 设置confirm回调函数
    template.setConfirmCallback(new RabbitTemplate.ConfirmCallback() {
    @Override
    public void confirm(CorrelationData correlationData, boolean b, java.lang.String s) {
    if (b) System.out.println("消息发送成功");
    else System.out.println("消息发送失败");
    } });
    // 模拟生产者发送信息
    // 不存在的交换机--异常情况
    template.convertAndSend("sb_topic_exchange_noexist","user.info","日志级别:info;日志模块:user;日志信息:*****");
    } @Test
    void testReturnFalse() {
    // 设置return回调函数
    template.setReturnCallback(new RabbitTemplate.ReturnCallback() {
    @Override
    public void returnedMessage(Message message, int i, java.lang.String s, java.lang.String s1, java.lang.String s2) {
    System.out.println(message.toString());
    System.out.println(s+"*********");
    } });
    template.setMandatory(true);
    // 模拟生产者发送信息
    // 正确的交换机 错误的routekey -- 异常情况
    template.convertAndSend("sb_topic_exchange","noexist.user.info","日志级别:info;日志模块:user;日志信息:*****");
    }
    }

消费者确认

重点在于消费者的下面两个方法

  • channel.basicAck 消费者签收
  • channel.basicNAck 消费者拒绝签收
  1. 开启消费者确认模式

    spring:
    rabbitmq:
    host: localhost
    port: 5672
    virtual-host: /
    username: root
    password: root
    # 设置消费端手动签收
    listener:
    direct:
    acknowledge-mode: manual
    simple:
    acknowledge-mode: manual
  2. 创建消费者

    /**
    * 注入消费者--手动签到
    */
    @Component
    @RabbitListener(queues = "sb_topic_queue1")
    public class Consumer2 { @RabbitHandler
    public void testComsumer(String msg, Channel channel, Message message) throws InterruptedException, IOException {
    // 消费端设置手动签收代码
    try {
    System.out.println(msg);
    // 正常签收,mq收到此消息被正常签收后即可从队列中删除vi信息
    // 是哟了那个channel的方法
    // 第一个参数是deliverytag 标识哪条信息 第二个参数是是否批量签收
    // int i=2/0; 模拟异常
    channel.basicAck(message.getMessageProperties().getDeliveryTag(),true);
    System.out.println("消费者签收了该信息,服务器你可以删了");
    }catch (Exception e){
    // 异常拒绝签收,让mq重发此信息
    System.out.println("该信息丢了,给我重发");
    channel.basicNack(message.getMessageProperties().getDeliveryTag(),true,true);
    // 该信息丢了,但是不需要你重发
    // channel.basicNack(message.getMessageProperties().getDeliveryTag(),true,false);
    }
    }
    }
  3. 创建生产者

    @SpringBootTest
    class RabiitmqSpringbootApplicationTests { @Autowired
    RabbitTemplate template; @Test
    void testConsumerAck() {
    template.convertAndSend("sb_topic_exchange","noexist.user.info","日志级别:info;日志模块:user;日志信息:*****");
    }
    }

参考

消息队列RabbitMQ(三):消息确认机制的更多相关文章

  1. RabbitMQ消息发布和消费的确认机制

    前言 新公司项目使用的消息队列是RabbitMQ,之前其实没有在实际项目上用过RabbitMQ,所以对它的了解都谈不上入门.趁着周末休息的时间也猛补习了一波,写了两个窗体应用,一个消息发布端和消息消费 ...

  2. 消息队列——RabbitMQ学习笔记

    消息队列--RabbitMQ学习笔记 1. 写在前面 昨天简单学习了一个消息队列项目--RabbitMQ,今天趁热打铁,将学到的东西记录下来. 学习的资料主要是官网给出的6个基本的消息发送/接收模型, ...

  3. ASP.NET Core消息队列RabbitMQ基础入门实战演练

    一.课程介绍 人生苦短,我用.NET Core!消息队列RabbitMQ大家相比都不陌生,本次分享课程阿笨将给大家分享一下在一般项目中99%都会用到的消息队列MQ的一个实战业务运用场景.本次分享课程不 ...

  4. 消息队列rabbitmq rabbitMQ安装

    消息队列rabbitmq   12.1 rabbitMQ 1. 你了解的消息队列 生活里的消息队列,如同邮局的邮箱, 如果没邮箱的话, 邮件必须找到邮件那个人,递给他,才玩完成,那这个任务会处理的很麻 ...

  5. 消息队列RabbitMQ的安装配置与PHP中的使用

    一.RabbitMQ安装 windows安装 下载地址: https://github.com/rabbitmq/rabbitmq-server/releases/download/v3.8.3/ra ...

  6. RabbitMQ(三):消息持久化策略

    原文:RabbitMQ(三):消息持久化策略 一.前言 在正常的服务器运行过程中,时常会面临服务器宕机重启的情况,那么我们的消息此时会如何呢?很不幸的事情就是,我们的消息可能会消失,这肯定不是我们希望 ...

  7. SpringBoot(八) Spring和消息队列RabbitMQ

    概述 1.大多数应用中,可以通过消息服务中间件来提升系统异步能力和拓展解耦能力. 2.消息服务中的两个重要概念:消息代理(Message broker)和目的地(destination) 当消息发送者 ...

  8. (一)RabbitMQ消息队列-RabbitMQ的优劣势及产生背景

    原文:(一)RabbitMQ消息队列-RabbitMQ的优劣势及产生背景 本篇并没有直接讲到技术,例如没有先写个Helloword.我想在选择了解或者学习一门技术之前先要明白为什么要现在这个技术而不是 ...

  9. 消息队列rabbitmq/kafka

    12.1 rabbitMQ 1. 你了解的消息队列 rabbitmq是一个消息代理,它接收和转发消息,可以理解为是生活的邮局.你可以将邮件放在邮箱里,你可以确定有邮递员会发送邮件给收件人.概括:rab ...

  10. 消息队列rabbitmq的五种工作模式(go语言版本)

    前言:如果你对rabbitmq基本概念都不懂,可以移步此篇博文查阅消息队列RabbitMQ 一.单发单收 二.工作队列Work Queue 三.发布/订阅 Publish/Subscribe 四.路由 ...

随机推荐

  1. js_笔记_8月7日记录_活动对象_作用域链_按值传递

    活动对象:简单说就是这个函数的参数和显示声明的变量或函数. 函数内接受的参数实际是创建了一个局部变量:[形参名] = [传进来的值],js的函数传参只传值. 作用域链:执行流进入一个函数,会先创建出作 ...

  2. PTA 单链表分段逆转

    6-9 单链表分段逆转 (25 分)   给定一个带头结点的单链表和一个整数K,要求你将链表中的每K个结点做一次逆转.例如给定单链表 1→2→3→4→5→6 和 K=3,你需要将链表改造成 3→2→1 ...

  3. time模块&datetime模块

    import time a=time.localtime(time.time()) #将时间戳转换为当前时区的元组 print(a) c=time.gmtime(time.time()) #把时间戳转 ...

  4. java例题_23 递归求年龄

    1 /*23 [程序 23 求岁数] 2 题目:有 5 个人坐在一起,问第五个人多少岁,他说比第 4 个人大 2 岁.问第 4 个人岁数,他说比第 3 个 3 人大 2 岁.问第三个人,又说比第 2 ...

  5. OpenCV 之 平面单应性

    上篇 OpenCV 之 图象几何变换 介绍了等距.相似和仿射变换,本篇侧重投影变换的平面单应性.OpenCV相关函数.应用实例等. 1  投影变换 1.1  平面单应性 投影变换 (Projectiv ...

  6. 第21 章 : Kubernetes 存储架构及插件使用

    Kubernetes 存储架构及插件使用 本文将主要分享以下三方面的内容: Kubernetes 存储体系架构: Flexvolume 介绍及使用: CSI 介绍及使用. Kubernetes 存储体 ...

  7. 第14 章 : Kubernetes Service讲解

    Kubernetes Service 本文将主要分享以下四方面的内容: 为什么需要 K8s service: K8s service 用例解读: K8s service 操作演示: K8s servi ...

  8. 【Azure 事件中心】EPH (EventProcessorHost) 消费端观察到多次Shutdown,LeaseLost的error信息,这是什么情况呢?

    问题详情 使用EPH获取Event Hub数据时,多次出现连接shutdown和LeaseLost的error  ,截取某一次的error log如: Time:2021-03-10 08:43:48 ...

  9. 如何从 dump 文件中提取出 C# 源代码?

    一:背景 相信有很多朋友在遇到应用程序各种奇葩问题后,拿下来一个dump文件,辛辛苦苦分析了大半天,终于在某一个线程的调用栈上找到了一个可疑的方法,但 windbg 常常是以 汇编 的方式显示方法代码 ...

  10. 201871010130-周学铭 实验三 结对项目—《D{0-1}KP 实例数据集算法实验平台》项目报告

    项目 内容 课程班级博客链接 18卓越班 这个作业要求链接 实验三结对编程要求 我的课程学习目标 体验软件项目开发中的两人合作,练习结对编程(Pair programming).掌握Github协作开 ...