PHP使用RabbitMQ入门3——发布/订阅

发布/订阅

通过前两篇文章,我们了解到生产了消息后,一旦某个消费者消费了消息,那么其他消息者就无法消费该消息了,但有的时候,我们需要把消息发送到每一个消费者。例如我们有两个消费者,一个用于给用户发送短信,一个用于给用户发送邮件,但消息内容都是一样的。

分发一个消息给多个消费者(consumers)。这种模式被称为“发布/订阅”。

交换机(Exchanges)

RabbitMQ消息模型的核心理念是:发布者(producer)不会直接发送任何消息给队列。事实上,发布者(producer)甚至不知道消息是否已经被投递到队列。

发布者(producer)只需要把消息发送给一个交换机(exchange)。交换机一边从发布者方接收消息,一边把消息推送到队列。交换机必须知道如何处理它接收到的消息,是应该推送到指定的队列还是是多个队列,或者是直接忽略消息。这些规则是通过交换机类型(exchange type)来定义的。

有几个可供选择的交换机类型:直连交换机(direct), 主题交换机(topic), (头交换机)headers和 扇型交换机(fanout)。我们在这里主要说明最后一个 —— 扇型交换机(fanout)。先创建一个fanout类型的交换机,命名为logs:

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

扇型交换机(fanout)会把消息发送给它所知道的所有队列。

交换器列表

rabbitmqctl能够列出服务器上所有的交换器:

rabbitmqctl list_exchanges

匿名的交换器

前面的文章中我们对交换机一无所知,但仍然能够发送消息到队列中。因为我们使用了命名为空字符串(“”)默认的交换机。
例如之前是这样发布一则消息:

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

这里我们使用默认或者匿名交换机:消息将会根据指定的routing_key分发到指定的队列,routing key是basic_publish函数的第三个参数。

发送消息到一个已命名的交换机:

$channel->exchange_declare('msgs', 'fanout', false, false, false);   //msgs就是交换机的名称
$channel->basic_publish($msg, 'msgs');

临时队列

在 php-amqplib 客户端,当我们提供队列名称为空字符串时,我们创建了一个具有生成名称的非持久队列:

list($queue_name, ,) = $channel->queue_declare("");

方法返回时,$queue_name变量包含一个随机生成的RabbitMQ队列名称。例如,类似amq.gen-jzty20brgko-hjmujj0wlg。

绑定(Bindings)

创建了队列与交换机后,还需要完成交换机与队列的绑定,这样交换机才能知道把消息发送到哪个队列。

$channel->queue_bind($queue_name, 'msgs');

绑定后,交换机将会把消息添加到指定的队列中。

绑定(binding)列表

通过以下命令可以列出所有现存的绑定:

rabbitmqctl list_bindings

新建生产者:send_msg.php

新建send_msg.php用于发送消息至指定的交换机,代码如下:

<?php

require_once __DIR__ . '/vendor/autoload.php';
use PhpAmqpLib\Connection\AMQPStreamConnection;
use PhpAmqpLib\Message\AMQPMessage;

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

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

$data = implode(' ', array_slice($argv, 1));
if(empty($data)) $data = "nobody";
$msg = new AMQPMessage($data);

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

echo "Sent message to ", $data, "\n";

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

?>

新建消费者:rec_msg1.php

新建rec_msg1.php用于将消息以email的方式发送给指定用户。

<?php

require_once __DIR__ . '/vendor/autoload.php';
use PhpAmqpLib\Connection\AMQPStreamConnection;

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

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

list($queue_name, ,) = $channel->queue_declare("", false, false, true, false);

$channel->queue_bind($queue_name, 'msgs');

echo 'Waiting for messages. To exit press CTRL+C', "\n";

$callback = function($msg){
  echo 'Send email to ', $msg->body, "\n";
};

$channel->basic_consume($queue_name, '', false, true, false, false, $callback);

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

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

?>

新建消费者:rec_msg2.php

新建rec_msg2.php用于将消息以短信方式发送给指定用户。

<?php

require_once __DIR__ . '/vendor/autoload.php';
use PhpAmqpLib\Connection\AMQPStreamConnection;

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

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

list($queue_name, ,) = $channel->queue_declare("", false, false, true, false);

$channel->queue_bind($queue_name, 'msgs');

echo 'Waiting for messages. To exit press CTRL+C', "\n";

$callback = function($msg){
  echo 'Send sms to ', $msg->body, "\n";
};

$channel->basic_consume($queue_name, '', false, true, false, false, $callback);

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

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

?>

简单说明一下,目前新建了3个文件,send_msg.php文件为消息生产者,用于将要发送给用户的消息推送到rabbitmq服务器,rec_msg1.php和rec_msg2.php为消费者,其中rec_msg1.php用于将消息以email的方式发送给用户,rec_msg2.php则用于将消息以短信的方式发送给用户。

运行以上三个文件后,效果如下图:

通过运行结果可以看出,生产者(send_msg.php)将消息推送到rabbimq后,rabbitmq将消息分发到了每一个生产者。

参考

RabbitMQ手册【发布/订阅】

发表评论

邮箱地址不会被公开。 必填项已用*标注