【问题标题】:RabbitMQ: re-consume non-acked message in fanout exchangeRabbitMQ:在扇出交换中重新使用未确认的消息
【发布时间】:2017-10-19 19:22:15
【问题描述】:

这是我的问题。我想以Publish/Subscribe 方式创建一个玩具应用程序,我以那个为例。但我想通过确认来实现我的消费者。我期望以下行为:在我的一个消费者死亡后,我再次在控制台中运行该脚本,然后处理消息。

这是发布商的代码:

<?php

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

use PhpAmqpLib\Connection\AMQPStreamConnection;
use PhpAmqpLib\Message\AMQPMessage;

$connection = new AMQPStreamConnection('localhost', 5672, 'guest', 'guest');
$channel = $connection->channel();

$channel->exchange_declare('logs', 'fanout', false, true, false);

$data = implode(' ', array_slice($argv, 1));
if(empty($data)) $data = "info: Hello World!";
$msg = new AMQPMessage($data);

$channel->basic_publish($msg, 'logs');

echo " [x] Sent ", $data, "\n";

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

这是订阅者的代码:

<?php

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

use PhpAmqpLib\Connection\AMQPStreamConnection;

$connection = new AMQPStreamConnection('localhost', 5672, 'guest', 'guest');
$channel = $connection->channel();

$channel->exchange_declare('logs', 'fanout', false, true, false);

$channel->queue_declare("queue1", false, true, true, false);

$channel->queue_bind("queue1", 'logs');

echo ' [*] Waiting for logs. To exit press CTRL+C', "\n";

$callback = function($msg){
    echo ' [x] ', $msg->body, "\n";
    die();
    $msg->delivery_info['channel']->basic_ack($msg->delivery_info['delivery_tag']);
};

$channel->basic_consume("queue1", '', false, false, false, false, $callback);

while(count($channel->callbacks)) {
    $channel->wait();
}

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

当我运行它时,[x] info: Hello World! 显示在订阅者的控制台中,比我预期的消息没有被确认,因为脚本只是死了。再次运行后,没有消息消耗。而且 rabbitMQ 网络面板也显示没有未确认的消息。

我做错了什么?

【问题讨论】:

    标签: php rabbitmq


    【解决方案1】:

    那是因为您声明了 exclusive queue。一旦连接丢失,它就会被删除。因此,当您按下 Ctrl + C 时,队列被删除,当您再次运行该脚本时,会出现一个新队列,其中没有任何消息。所以你需要声明一个非独占队列:

    $channel->queue_declare("queue1", false, true, false, false);
    

    【讨论】:

      最近更新 更多