Rabbit MQ 死信队列、延迟队列

什么是死信队列

消息变成死信的三种情况:

  • 消息被拒绝(basic.reject / basic.nack),并且requeue = false
  • 消息TTL过期
  • 队列达到最大长度

“死信”是RabbitMQ中的一种消息机制,当消息出现以上三种情况,就会变成死信,死信”消息会被RabbitMQ进行特殊处理,如果配置了死信队列信息,那么该消息将会被丢进死信队列中,如果没有配置,则该消息将会被丢弃。


死信队列示意图

什么是延迟队列

延迟队列就是利用TTL和死信队列来实现的,在过期时间后将消息转发到死信队列,在死信队列做逻辑处理。从上图中可以看出,如果普通消费者收到消息后啥也不干,等到了过期时间消息就会转发到死信队列中,相当于,消息只是在普通消费者那里暂存了TTL时间,然后交给了死信队列,也就实现了延时效果。

延迟队列使用场景

  • 订单超时未支付自动关闭
  • 召回N天前注册的用户
  • 订单购买成功N分钟后检查下游环节是否正常,如开通会员权益

如何配置死信队列

在业务队列中配置死信队列信息

    'x-dead-letter-exchange' => '',//死信交换机
    'x-dead-letter-routing-key' => '',//死信路由key
    'x-message-ttl' => '',//超时时间,单位毫秒 
    'x-max-length' => '',//队列最大长度  

如何让消息过期

让消息过期有2种方式,一种是在队列中设置x-message-ttl

//声明支付通知队列
$channel->queue_declare($queue_pay_notice,false,false,false,false,false,new \PhpAmqpLib\Wire\AMQPTable([
    'x-dead-letter-exchange'=>$exchange_order_delay,
    'x-dead-letter-routing-key'=>$routing_key_order_delay,
    'x-message-ttl'=>10000,//过期时间10S
]));

这种设置方式会让每个消息的过期时间都一样,都是10S,如果想灵活的设置过期时间,可以再发送消息的时候设置expiration

$msg = new \PhpAmqpLib\Message\AMQPMessage('hello',['expiration'=>10000]) //10s后过期;
 $channel->basic_publish(msg, $exchange_name, $routing_key);

如何实现死信队列

普通消费者

死信队列在普通消费者中的代码需要处理的工作是最多的,需要声明普通队列和死信队列,普通交换机和死信交换机,并绑定普通交换机的路由键,死信队列的路由键,并在普通队列中配置死信队列的交换机和路由键。

生产者

生产者还是一如既往只要负责发消息即可。生产者并不知道消息最终会进入什么流程,他只管把消息发送到对应的普通交换机即可。

死信队列消费者

死信队列消费者也不需要关心之前的逻辑,只要将收到的消息做对应的处理即可。

PHP demo

生产者 send.php

<?php
require_once __DIR__ . '/../../vendor/autoload.php';

$queue_order = 'queue_order';
$queue_delay_order = 'queue_delay_order';
$exchange_delay_order = 'exchange_delay_order';
$routing_delay_order = 'routing_delay_order';
$delay_ttl = 10 * 1000;
//场景 关闭10分钟后未支付的订单
$conn = new \PhpAmqpLib\Connection\AMQPStreamConnection('localhost', '5672', 'guest', 'guest');
$channel = $conn->channel();

$channel->queue_declare($queue_order, false, false, false, false, false, New \PhpAmqpLib\Wire\AMQPTable([
    'x-dead-letter-exchange' => $exchange_delay_order,
    'x-dead-letter-routing-key' => $routing_delay_order,
    'x-message-ttl' => $delay_ttl
]));

//绑定死信队列
$channel->queue_declare($queue_delay_order, false, false, false, false);
$channel->exchange_declare($exchange_delay_order, 'direct', false, false, false);
$channel->queue_bind($queue_delay_order, $exchange_delay_order, $routing_delay_order);
for ($i = 1; $i <= 10; $i++) {
    $msg = json_encode(['orderId' => $i,'time'=>date('H:i:s')]);
    echo 'seng msg' . $msg . PHP_EOL;
    $msg = new \PhpAmqpLib\Message\AMQPMessage($msg);
    $channel->basic_publish($msg, '',$queue_order);
}

$channel->close();
$conn->close();

业务消费者 receive.php

<?php
require_once __DIR__ . '/../../vendor/autoload.php';

$queue_order = 'queue_order';
//场景 关闭10分钟后未支付的订单
$conn = new \PhpAmqpLib\Connection\AMQPStreamConnection('localhost', '5672', 'guest', 'guest');
$channel = $conn->channel();

$channel->basic_consume($queue_order, '', false, false, false, false, function ($msg) {
    $data = json_decode($msg->body, true);

    if ($data['orderId'] % 3 == 0) {//3,6,9
       //确认消息
        $msg->delivery_info['channel']->basic_ack($msg->delivery_info['delivery_tag']);//ack
        echo 'get msg & ack orderId:' . $data['orderId'] . $msg->body . PHP_EOL;
    } else if ($data['orderId'] % 3 == 1) {//1,4,7,10
       //不确认消息
        $msg->delivery_info['channel']->basic_nack($msg->delivery_info['delivery_tag']);//nack 
        echo 'get msg & nack orderId:' . $data['orderId'] . $msg->body . PHP_EOL;
    } else {//2,5,8
       //拒绝消息
        $msg->delivery_info['channel']->basic_reject($msg->delivery_info['delivery_tag'], false);
        echo 'get msg & reject orderId:' . $data['orderId'] . $msg->body . PHP_EOL;
    }

});

while ($channel->is_consuming()) {
    $channel->wait();
}
$channel->close();
$conn->close();

延迟队列消费者 delay_receive.php

<?php
require_once __DIR__ . '/../../vendor/autoload.php';

$queue_delay_order = 'queue_delay_order';
$conn = new \PhpAmqpLib\Connection\AMQPStreamConnection('localhost', '5672', 'guest', 'guest');
$channel = $conn->channel();

$channel->queue_declare($queue_delay_order, false, false, false, false);
$channel->basic_consume($queue_delay_order, '', false, false, false, false, function ($msg) {
    echo date('H:i:s').' get msg ' . $msg->body . PHP_EOL;
    $msg->delivery_info['channel']->basic_ack($msg->delivery_info['delivery_tag']);//确认收到消息
});

while ($channel->is_consuming()) {
    $channel->wait();
}
$channel->close();
$conn->close();
receive.php执行结果
delay_receive.php执行结果

可以看到 nack和reject的消息进入到了死信队列中。如果没有运行receive.php,那么消息将会在设置的$delay_ttl后进入死信队列。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 212,884评论 6 492
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,755评论 3 385
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 158,369评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,799评论 1 285
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 65,910评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,096评论 1 291
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,159评论 3 411
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,917评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,360评论 1 303
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,673评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,814评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,509评论 4 334
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,156评论 3 317
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,882评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,123评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,641评论 2 362
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,728评论 2 351

推荐阅读更多精彩内容