消息确认机制

前言


消息队列的下游,业务逻辑可能复杂,处理任务可能花费很长时间。若在一条消息到达它的下游,任务刚处理了一半,由于不确定因素,下游的任务处理进程

被kill掉啦,导致任务无法执行完成。而沿用我们前面几章的消息删除【消息一旦抛给下游,就立马从队列删除】,这可能会引发问题——消息没有处理完,但是队列

里的消息已经被删除了。

因此,rabbitmq内含 消息确认机制【Message acknowledgment】,简称ack。rabbitmq将消息发送给consumer,此刻消息不会从队列删除,consumer消费完毕消息后,

给rabbitmq发送一条ack,在rabbitmq收到ack后,才从队列里删除。

代码


在consumer层的代码,需要修改两步。

一、添加ack代码

  1. $msg->delivery_info['channel']->basic_ack($msg->delivery_info['delivery_tag']);

二、开启消息确认机制

  1. $channel->basic_consume('task_queue', '', false, false, false, false, $callback); //basic_consume第四个参数置成false,no_ack=false

整体代码见下

生产者sender.php

  1. <?php
  2. /**
  3. * sender.php
  4. * Created by PhpStorm.
  5. * User: wangdaxi
  6. * Date: 2017/10/18
  7. * Time: 14:26
  8. */
  9. require_once __DIR__ . '/vendor/autoload.php';
  10. use PhpAmqpLib\Connection\AMQPStreamConnection;
  11. use PhpAmqpLib\Message\AMQPMessage;
  12.  
  13. $connection = new AMQPStreamConnection('127.0.0.1', 5672, 'guest', 'guest');
  14. $channel = $connection->channel();
  15.  
  16. $channel->queue_declare('hello', false, false, false, false);
  17.  
  18. $data = implode(" ", array_slice($argv, 1));
  19. empty($data) && $data = "Hello World!";
  20.  
  21. $msg = new AMQPMessage($data);
  22.  
  23. $channel->basic_publish($msg, '', 'hello');
  24.  
  25. echo " [x] Sent '$data'\n";
  26.  
  27. //close the channel and connection;
  28. $channel->close();
  29. $connection->close();

消费者receive.php

  1. <?php
  2. /**
  3. * receive.php
  4. * Created by PhpStorm.
  5. * User: wangdaxi
  6. * Date: 2017/10/18
  7. * Time: 14:34
  8. */
  9. require_once __DIR__ . '/vendor/autoload.php';
  10. use PhpAmqpLib\Connection\AMQPStreamConnection;
  11.  
  12. $connection = new AMQPStreamConnection('127.0.0.1', 5672, 'guest', 'guest');
  13. $channel = $connection->channel();
  14.  
  15. $channel->queue_declare('hello', false, false, false, false);
  16. echo ' [*] Waiting for messages. To exit press CTRL+C', "\n";
  17.  
  18. $callback = function($msg) {
  19. echo "[x] Received ", $msg->body, "\n";
  20. sleep(substr_count($msg->body, '.'));
  21. echo "[x] Done\n";
  22. //消息确认
  23. $msg->delivery_info['channel']->basic_ack($msg->delivery_info['delivery_tag']);
  24. };
  25. $channel->basic_consume('hello', '', false, false, false, false, $callback);
  26.  
  27. while(count($channel->callbacks)) {
  28. $channel->wait();
  29. }

验证


模拟消息下游进程被kill掉。

1. 开启两个终端分别作为消费者,记作C1,C2。分别等待接收消息。

2. 开启一个终端作为生产者。记作P。发送消息,用圆点来模拟耗时任务。

3. 发现C1接收到了消息,这里我们用 Ctrl+C关闭掉该进程。

4. 由于C1没有回传ack,且已中断,C2接收到了该消息,处理并且ack回传。

5. 通过 rabbitmq命令 查看hello队列messages_ready 和 messages_unacknowledged,ready状态的消息和未确认消息都是0。证明该消息已经被消费。

疑问


如果消费者执行完毕消息但是忘记了ack,会发生什么?

1. 修改消费者代码,注释掉ack代码。

2. 开启两个终端,分别作为消费者C1、生产者P。

3. 生产者P生产消息

4. 消费者接收到消息并处理

5. 命令查看消息数,发现未ack消息有两条。

6. 修改代码,解除ack通知的注释,另开终端作为消费者C2。发现C2不会接收到任何消息。

7. 关闭C1,C2接收C1未ack消息并处理。

8. 命令行查看发现消息都被消费掉。

结论


当消费者开启ack确认机制却忘记在处理完消息后回传rabbitmq ack时,会产生严重后果。

1. 未ack的消费者可以继续接收消息,但是不回传ack。导致rabbitmq内存被unacknowledged messages占用过多。

2. 从代码层面要解决掉,需要放弃掉这个进程,另开进程添加ack代码,进行消息回传。从而清除掉占用内存的已经被处理却未被删除的消息。

以上。

RabbitMq初探——消息确认的更多相关文章

  1. RabbitMq初探——消息均发

    消息均发 前言 由前文 RabbitMq初探——消息分发 可知,rabbitmq自带分发机制——消息会按顺序的投放到该队列下的多个消费者,例如1,3,5投放消费者C1,2,4,6投放消费者C2. 这就 ...

  2. Java使用RabbitMQ之消息确认(confirm模板)

    RabbitMQ生产者消息确认Confirm模式,分为普通模式.批量模式和异步模式,本次举例为普通模式. 源码: package org.study.confirm4; import com.rabb ...

  3. RabbitMQ的消息确认机制

    一:确认种类 RabbitMQ的消息确认有两种. 一种是消息发送确认.这种是用来确认生产者将消息发送给交换器,交换器传递给队列的过程中,消息是否成功投递.发送确认分为两步,一是确认是否到达交换器,二是 ...

  4. RabbitMq之消息确认

    最近阅读了rabbitmq的官方文档,然后结合之前面试时被问到关于消息队列的问题来探索一下关于消息队列的消息确认机制. 其实消息确认就是消费者确认消息被消费了, 生产者确认消息已经发送到了消息队列中了 ...

  5. RabbitMQ学习笔记六:RabbitMQ之消息确认

    使用消息队列,必须要考虑的问题就是生产者消息发送失败和消费者消息处理失败,这两种情况怎么处理. 生产者发送消息,成功,则确认消息发送成功;失败,则返回消息发送失败信息,再做处理. 消费者处理消息,成功 ...

  6. RabbitMq初探——消息持久化

    消息持久化 前言 通过上一节,我们知道,有消息确认机制,保证了当消费者进程挂掉后,消息的不丢失. 但是如果rabbitmq挂掉呢?它的队列和消息都会丢失的.为了保证消息在rabbitmq挂掉重启后不丢 ...

  7. RabbitMQ 之消息确认机制(事务+Confirm)

    概述 在 Rabbitmq 中我们可以通过持久化来解决因为服务器异常而导致丢失的问题,除此之外我们还会遇到一个问题:生产者将消息发送出去之后,消息到底有没有正确到达 Rabbit 服务器呢?如果不错得 ...

  8. RabbitMQ (十一) 消息确认机制 - 消费者确认

    由于生产者和消费者不直接通信,生产者只负责把消息发送到队列,消费者只负责从队列获取消息(不管是push还是pull). 消息被"消费"后,是需要从队列中删除的.那怎么确认消息被&q ...

  9. springboot整合rabbitmq,支持消息确认机制

    安装 推荐一篇博客https://blog.csdn.net/zhuzhezhuzhe1/article/details/80464291 项目结构 POM.XML <?xml version= ...

随机推荐

  1. 最值得学习阅读的10个C语言开源项目代码

    阅读优秀代码是提高开发人员修为的一种捷径-- 1. Webbench Webbench是一个在linux下使用的非常简单的网站压测工具.它使用fork()模拟多个客户端同时访问我们设定的URL,测试网 ...

  2. cannot nest '/dubboService/src/main/resources' inside '/dubboService/src/main' .To enable the nesting exclude '/resources' from '/dubboService/src/main'

    eclipse Maven--->update Project时出现以上错误: cannot nest '/dubboService/src/main/resources' inside '/d ...

  3. linux 正则表达式

    redis-cli INFO|tr -d '\r'|egrep -v '^(#.*)?$'|sed -E 's/^([^:]*):(.*)$/redisInfo[\1]="\2"/ ...

  4. awk使用

    [awk使用] 例:awk -F ':' '{print $1"\t"$7}' 参考:http://www.cnblogs.com/ggjucheng/archive/2013/0 ...

  5. Source命令及脚本的执行方式

    [Source命令及脚本的执行方式] source filename 与 sh filename 及./filename执行脚本的区别在那里呢? 1.当shell脚本具有可执行权限时,用sh file ...

  6. s 销售视图数据

    INSERT,需将公司ID匹对到以下EXCEL表 INSERT,需将公司ID匹对到以下EXCEL表 [Public] ConnectString=host="siebel://10.10.0 ...

  7. 读取指定路径的Properties文件

    1.读取项目内的properties文件,项目内的properties文件一般都放在resource文件夹下面, 通过getClassLoader().getResourceAsStream()来获取 ...

  8. 14.Longest Common Prefix (String)

    Write a function to find the longest common prefix string amongst an array of strings. class Solutio ...

  9. 微信小程序通讯录首字母索引效果,车辆品牌选择列表

    效果图: wxml代码: <block wx:for="{{list}}"> <view class='letter' id="letter{{inde ...

  10. QEMU 代码分析:BIOS 的加载过程

    http://www.ibm.com/developerworks/cn/linux/1410_qiaoly_qemubios/ QEMU 中使用 BIOS 简介 BIOS 提供主板或者显卡的固件信息 ...