php消费rabbitmq消息QoS,中间件系列十 RabbitMQ之消费者端的消息确认机制

概述

在RabbitMQ中,即使将queue,exchange, message等都设置了持久化之后,还是不能保证100%保证数据不丢失了。为了实现消息不丢失,我们需要从Consumer端和Productor端同时进行处理。本篇文章先介绍Consumer端,在AMPQ-0-9-1中有定义从消费者到RabbitMQ的消息确认机制,通过此机制可以保证消息能够从RabbitMQ正确到达消费者端。本文介绍在RabbitMQ中如何实现消费者端的消息确认机制,包括如下内容

1 消费者的实现机制

2 消费者端的代码实现

3 使用wireshark对消息确认的关键包进行转包,并进行分析

4 在使用消息确认机制的注意点

消费者端投递确认机制

在消费者端确认的方式

RabbitMQ中的两种确认方式:1 自动确认方式:RabbitMQ成功将消息发出(即将消息成功写入TCP Socket)中立即认为本次投递已经被正确处理,不管消费者端是否成功处理本次投递

2 手动处理方式:消费者收到消息后,手动调用basic.ack/basic.nack/basic.reject后,RabbitMQ收到这些消息后,才认为本次投递成功

批量手动投递确认

消息手动除了一次确认一条,也可以一次确认多条。为了减少网络流量,可以批量手动确认。在应答时,设置basic.nack的multiple 字段为true,可以同时对delivery_tag和比delivery_tag值小的投递消息进行确认

例如,假设在通道上没有确认消息的delivery_tag是5,6,7和8,当basic.nack中delivery_tag被设置为8并且multiple 被设置为true时,方法执行成功后,从5到8的所有消息将被确认。 如果multiple 设置为false,那么交货5,6和7仍然是未确认的。

投递唯一码: Delivery Tags

当消费者向RabbitMQ注册后,RabbitMQ使用basic.deliver向消费者投递消息时,消息体上会带上delivery tag,这个值会唯一标识本次投递,在同一通道上,此值是唯一的。delivery tag值有64位长度,值从1开始,每发送一次消息值递增1,最大值为9223372036854775807。消费者端在应答消息时,带上此参数,告诉RabbitMQ某次投递已经正确应答。

消费者端消息投递确认代码

工程:

工程:

代码路径:

消费者代码:

代码:关键点

a. channel.basicConsume设置接收非自动确认

b. 在处理完消息后,调用channel.basicAck进行手动消息确认// 默认消费者实现

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(" [ConsumerConfirmRecv] Received '" + message + "'");

// 消息正向确认

channel.basicAck(envelope.getDeliveryTag(),true);

// 消息否定确认: 如果设置multiple=false,requeue值启作用,如果设置multiple=true,则requeue无论设置什么值,后台统一处理成true

// channel.basicNack(envelope.getDeliveryTag(),false, false);

}

};

// 接收消息:设置非自动确认

channel.basicConsume(QUEUE_NAME, false, consumer);

消费者端抓包分析

执行以上的代码,并进行抓包分析其中关键包。和一般的消息包不同点时,一般的消费者端的包分析见中间件系列九 RabbmtiMQ 通过wireshark抓包学习AMQP协议

正向确认-Basic.ack

和一般的消息包不同点时,一般的消费者端的包分析见本文,只是多了Basic.ack包

4284

131-132帧 RabbitMQ向消费者推送消息,消费者端进行否定确认

131帧 RabbitMQ向消费者推送消息Frame 131: 269 bytes on wire (2152 bits), 269 bytes captured (2152 bits) on interface 0

Ethernet II, Src: PcsCompu_48:72:ad (08:00:27:48:72:ad), Dst: Giga-Byt_bf:ae:ce (40:8d:5c:bf:ae:ce)

Internet Protocol Version 4, Src: 10.240.80.147, Dst: 10.240.80.99

Transmission Control Protocol, Src Port: 5672, Dst Port: 49877, Seq: 600, Ack: 642, Len: 215

# 返回Consume的调用结果

Advanced Message Queueing Protocol

Type: Method (1)

Channel: 1

Length: 36

Class: Basic (60)

Method: Consume-Ok (21)

Arguments

# 此消息的消费者的编号

Consumer-Tag: amq.ctag-AhhpMhEMC5VuWcny42rObg

# 要投递的消息参数

Advanced Message Queueing Protocol

Type: Method (1)

Channel: 1

Length: 87

Class: Basic (60)

Method: Deliver (60)

Arguments

# 要投递的消息者标志

Consumer-Tag: amq.ctag-AhhpMhEMC5VuWcny42rObg

# 消息的编号,从1开始

Delivery-Tag: 1

# 是否是重新投递的消息,如果消息是第一次投递,则此值是false,如果此消息是重新投递的,则此值为true

.... ...0 = Redelivered: False

# 来源的交换机

Exchange: consumerconfirm-exchange

# 路由键

Routing-Key: consumer-confirm

# 要投递的消息的头

Advanced Message Queueing Protocol

Type: Content header (2)

Channel: 1

Length: 27

Class ID: Basic (60)

Weight: 0

Body size: 33

Property flags: 0x9800

# 消息的属性

Properties

Content-Type: text/plain

Delivery-Mode: 2

Priority: 0

# 要投递的消息的内容

Advanced Message Queueing Protocol

Type: Content body (3)

Channel: 1

Length: 33

Payload: 436f6e73756d6572436f6e6669726d53656e642131353137...

132帧 消费者端对消息的确认

4284

否定确认-Basic.Nack

和一般的消息包不同点时,一般的消费者端的包分析见本文,只是多了Basic.Nack包

4284

275-276帧 RabbitMQ向消费者推送消息,消费者端进行否定确认

275帧 RabbitMQ向消费者推送消息,和上节内容相似,这里略

276帧 消费者端进行否定确认,比basic.ack多了Requeue属性

4284

其他消费者端的为了保证正确处理数据的机制

在消费者端为了正确处理数据,还可以设置Qos和投递失败

通道预取设置(Channel Prefetch Setting (QoS))

只能在消息手动确认模式中启作用。

为了避免消费者端一次同时处理过多的消息,可以通过basic.qos设置最大的预取值。该值定义了通道上允许的最大未确认消息,一旦未确认消息的数量达到配置值,RabbitMQ将停止在通道上传送更多消息,直到至少有一个未被确认的消息被确认。

备注:通道预取设置在basic.get (“pull API”)中是不启作用,即使在消息手动确认模式中

消费者确认模式,预取和吞吐量

在RabbitMQ中影响吞吐量最大的参数是:消息确认模式和Qos预取值

自动消息确认模式或设置Qos预取值为无限虽然可以最大的提高消息的投递速度,但是在消费者端未及时处理的消息的数量也将增加,从而增加消费者RAM消耗,使用消费者端奔溃。所以以上两种情况需要谨慎使用。

RabbitMQ官方推荐Qos预取值设置在 100到300范围内的值通常提供最佳的吞吐量,并且不会有使消费者奔溃的问题

消费者失败或失去连接时:自动重新排队

在消息手动确认模式中,如果发生以下情况投递消息所有的通道或连接被突然关闭(包括消费者端丢失TCP连接、消费者应用程序(进程)挂掉、通道级别的协议异常)任何已经投递的消息但是没有被消费者端确认的消息会自动重新排队。

请注意,连接检测不可用客户端需要一段时间才会发现,所以会有一段时间内的所有消息会重新投递

因为消息的可能重新投递,所有必须保证消费者端的接口的幂等。

多次确认和对未知delivery_tag进行确认

如果消费者对同一delivery_tag进行多次确认,则抛出通道异常PRECONDITION_FAILED。如果对

未知的delivery_tag进行确认,也会抛出通道异常。

代码

--------------------- 本文来自 hry2015 的CSDN 博客 ,全文地址请点击:https://blog.csdn.net/hry2015/article/details/79416540?utm_source=copy

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

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值