tutorial:http://www.rabbitmq.com/tutorials/tutorial-two-java.html

这里解释接收消息端关于 acknowledge和prefetch的设置问题

这里有两段代码,sender,负责发送100条消息; recv,负责接收消息,每接收到一条消息sleep 1 秒。

 channel.queueDeclare(QUEUE_NAME, false, false, false, null);
//channel.basicQos(1,false);
System.out.println(" [*] Waiting for messages. To exit press CTRL+C"); Consumer consumer = new DefaultConsumer(channel) {
@Override
public void handleDelivery(String consumerTag, Envelope envelope, AMQP.BasicProperties properties, byte[] body)
throws IOException {
String message = new String(body, "UTF-8");
System.out.println(" [x] Received '" + message + "'");
try{
Thread.sleep(1000);
}catch (Exception e){ }
//channel.basicAck(envelope.getDeliveryTag(), false); }
};
channel.basicConsume(QUEUE_NAME, true, consumer);

先展示一个基础版本的recv。

运行这段代码遇到两个费解的现象

1. 先启动sender产生100条消息,后启动两个recv,发现所有消息都被第一个启动的recv接收了,第二个recv没有接收到任何消息。但如果先两个recv,再启动sender,那么两个recv是平均分配消息的。

2. 先启动sender,启动第一个recv,处理10条消息,关掉recv,再启动第二个recv,第二个recv没有消息。11~100条消息丢失了。

为了说明上面的问题,我们先描述一下一个消息在rmq server端的状态

在server端,消息进入queue后,首先是ready状态。如果有recv 消费这条消息,那么消息进入 unack 状态,当recv ack这条消息后,server端将删除消息。

现象1的解释:

先启动sender,server端有100个ready 消息。启动recv1,虽然recv1还没有来得及处理这些消息,但recv1 接收了100个消息。在server端100个消息都进入unack状态。因为我们设置的ack方式是autoack, line 20

因此所有消息立刻就从server端删除了。当我再启动recv2时,队列已经没有消息了,所以recv2没有接收到任何消息。反过来,先启动两个recv,sender生产消息的时候,server会按round robbin方式分配消息,因此两个recv各接收50条消息

现象2的解释:

因为消息都已经发送给了recv1,server端收到了ack,删除了消息,系统中只有recv1缓存了消息,如果关掉recv1,所有消息都会丢失。recv2无法拿到recv1未处理的消息。

下面我们看看改进版本。变化在于

  • line 20 autoack 改为false
  • line 16 每条消息处理后ack
 channel.queueDeclare(QUEUE_NAME, false, false, false, null);
//channel.basicQos(1,false);
System.out.println(" [*] Waiting for messages. To exit press CTRL+C"); Consumer consumer = new DefaultConsumer(channel) {
@Override
public void handleDelivery(String consumerTag, Envelope envelope, AMQP.BasicProperties properties, byte[] body)
throws IOException {
String message = new String(body, "UTF-8");
System.out.println(" [x] Received '" + message + "'");
try{
Thread.sleep(1000);
}catch (Exception e){ }
channel.basicAck(envelope.getDeliveryTag(), false); }
};
channel.basicConsume(QUEUE_NAME, false, consumer);

可以发现现象1 依旧,但现象2 不同了。我们可以通过命令查看server端的状态

rabbitmqctl list_queues name messages_unacknowledged messages_ready

启动sender后 unack=0,ready=100

启动recv1 后 unack=100, ready=0,所有消息都给了recv1

等recv1 处理几条消息后 unack=89, ready=0,recv1处理并ack了11条消息,

关闭recv1 后 unack=0, ready=85,未ack的消息都回到ready状态

启动recv2,unack=89, ready =0,所有消息都转给了recv2.

如果消息量很大,那么缓存消息就可能吃掉recv的所有内存导致系统崩溃。因此我们打开 line2。这样recv在ack一个消息后才会领取下一个消息。再来看看按照上面流程queue里消息的状态

启动sender后 unack=0,ready=100

启动recv1 后,unack=1, ready=99

启动recv2后,unack=2, ready=95

在消息处理完前,unack都是2,recv1和recv2 各持有一个消息

rabbitmq 消息的状态转换的更多相关文章

  1. SpringCloud之RabbitMQ消息队列原理及配置

    本篇章讲解RabbitMQ的用途.原理以及配置,RabbitMQ的安装请查看SpringCloud之RabbitMQ安装 一.MQ用途 1.同步变异步消息 场景:用户下单完成后,发送邮件和短信通知. ...

  2. RabbitMQ 消息确认机制

    消息确认机制 在之前异常处理部分就已经写了,对于consumer的异常退出导致消息丢失,可以时候consumer的消息确认机制.重复的就不说了,这里说一些不一样的. consumer的消息确认机制 当 ...

  3. (转)RabbitMQ消息队列(九):Publisher的消息确认机制

    在前面的文章中提到了queue和consumer之间的消息确认机制:通过设置ack.那么Publisher能不到知道他post的Message有没有到达queue,甚至更近一步,是否被某个Consum ...

  4. RabbitMQ消息队列(九):Publisher的消息确认机制

    在前面的文章中提到了queue和consumer之间的消息确认机制:通过设置ack.那么Publisher能不到知道他post的Message有没有到达queue,甚至更近一步,是否被某个Consum ...

  5. 使用EasyNetQ组件操作RabbitMQ消息队列服务

    RabbitMQ是一个由erlang开发的AMQP(Advanved Message Queue)的开源实现,是实现消息队列应用的一个中间件,消息队列中间件是分布式系统中重要的组件,主要解决应用耦合, ...

  6. RabbitMQ 消息顺序、消息幂等、消息重复、消息事务、集群

    1. 消息顺序 场景:比如下单操作,下单成功之后,会发布创建订单和扣减库存消息,但扣减库存消息执行会先于创建订单消息,也就说前者执行成功之后,才能执行后者. 不保证完全按照顺序消费,在 MQ 层面支持 ...

  7. RabbitMQ消息队列系列教程(二)Windows下安装和部署RabbitMQ

    摘要 本篇经验将和大家介绍Windows下安装和部署RabbitMQ消息队列服务器,希望对大家的工作和学习有所帮助! 目录 一.Erlang语言环境的搭建 二.RabbitMQ服务环境的搭建 三.Ra ...

  8. RabbitMQ消息队列(四)-服务详细配置与日常监控管理

    RabbitMQ服务管理 启动服务:rabbitmq-server -detached[ /usr/local/rabbitmq/sbin/rabbitmq-server -detached ] 查看 ...

  9. RabbitMQ 消息队列 应用

    安装参考    详细介绍   学习参考 RabbitMQ 消息队列 RabbitMQ是一个在AMQP基础上完整的,可复用的企业消息系统.他遵循Mozilla Public License开源协议. M ...

随机推荐

  1. Sa身份登陆SQL SERVER失败的解决方案

    经常使用windows身份登陆,久而久之,基本不动怎么用SQL SERVER身份验证登陆,所以趁着有空,就解决一下一些问题~~ 解决方案:  第一步:打开SSMS,先使用windows身份登陆,右击服 ...

  2. python excel操作 练习-#操作单列 #操作A到C列 #操作1到3行 #指定一个范围遍历所有行和列 #获取所有行 #获取所有列

    ##操作单列#操作A到C列#操作1到3行#指定一个范围遍历所有行和列#获取所有行#获取所有列 #coding=utf-8 from openpyxl import Workbook wb=Workbo ...

  3. win10如何设置自动睡眠时间(修改电源计划不好用的情况下)

    https://answers.microsoft.com/en-us/windows/forum/windows_10-power/windows-10-sleeping-when-set-not- ...

  4. linux chkconfig 管理服务开机自启动

    chkconfig命令主要用来更新(启动或停止)和查询系统服务的运行级信息.谨记chkconfig不是立即自动禁止或激活一个服务,它只是简单的改变了符号连接. 使用语法:chkconfig [--ad ...

  5. C++设计模式 之 “单一职责”模式:Decorator、Bridge

    part 1 “单一职责”模式 在软件组件的设计中,如果责任划分的不清晰,使用继承得到的结果往往是随着需求的变化,子类急剧膨胀,同时充斥着重复代码,这时候的关键是划清责任. 典型模式 Decorato ...

  6. 怎么解决深入学习PHP的瓶颈

    PHP给学习者的感觉是:初学的时候很容易,但是学了2-3年,就深刻感觉遇到了瓶颈,很难深入,放弃又可惜.所谓"鸡肋,食之无味弃之可惜"的感觉很是贴切. 经常会有这种感觉:不学,看似 ...

  7. 微信小程序——2、配置json文件

    配置文件详解 主配置文件app.json 主配置文件位于主目录中,用于进行全局配置.包括页面文件的路径.窗口表现.设置网络超时时间.设置多tab等 下面通过微信最初自带小程序来学习 { "p ...

  8. C#对两种类型动态库的使用

    一.托管:如果一个动态库本身是基于.NET的,那么可以直接在工程引用里右键添加引用,如微软的COM技术[因为你依托的是微软的框架,所以需要regsvr32注册] 二.非托管:如果不是基于.NEt的,那 ...

  9. 用Python为iOS和Android写跨平台的应用

    首先保证安装了最新的python(当前安装的是python3.6) 一.安装Kivy :python -m pip install --upgrade pip wheel setuptools pyt ...

  10. JavaScript 装饰者模式(this运用)

    例: function ConcreteClass() { this.performTask = function () { this.preTask(); console.log('doing so ...