php rabbitmq 订阅,五、消息订阅(Publish/Subscribe)

五、消息订阅(Publish/Subscribe)

由 学院君 创建于3年前, 最后更新于 7个月前

版本号 #2

9198 views

0 likes

0 collects

之前都是将消息发送到同一个 Consumer,而现在我们需将其发送到多个 Consumer。

我们将创建一个日志系统,它包含两个部分:第一个部分负责发出log(Producer),第二个部分负责接收并打印(Consumer)。我们将构建两个 Consumer,第一个将 log 写到物理磁盘上;第二个将 log 输出到屏幕。

"Fanout" not telling an exchange to distribute messages to different consumers, but to different queues. So, you need at least two queues binding to a "fanout" exchange. Then let your two consumers get message from those two queues, one consumer to one queue.

Exchange

Exchange 决定将 Message 发送到具体的 Queue,至于是发送给一个 Queue 还是多个 Queue,则需要通过 Exchange 的类型类决定。Exchange 分为三种类型:direct、topic 和 fanout。fanout 就是广播模式,会将 Message 都放到它所知道的所有 Queue 中:

$exchange->setType(AMQP_EX_TYPE_FANOUT);

现在我们可以直接通过 Exchange,而不需要 routing_key 来发送 Message 了:

$exchange->publish($message);

临时队列

截至现在,我们用的 Queue 都是有名字的。使用有名字的 Queue,使得在 Producer 和 Consumer 之前共享 Queue 成为可能。

在我们的日志系统中,不需要有名字的队列,要实现这个目标,需要在声明队列时不指定名称,而由系统随机分配:

$queue = new AMQPQueue($channel);

//$queue->setName($queueName);

$queue->setFlags(AMQP_EXCLUSIVE);

$queue->declareQueue();

//$queue->bind($exchangeName, $routeKey);

这时,通过 $queue->getName() 获取到的队列名称是随机生成的。

绑定

建立 Exchange 与 Queue 之间的绑定:

$queue->bind($exchangeName);

演示代码

emit_logs.php

/**

* 发送消息

*/

$exchangeName = 'logs';

$message = 'Hello World!';

// 建立TCP连接

$connection = new AMQPConnection([

'host' => 'localhost',

'port' => '5672',

'vhost' => '/',

'login' => 'guest',

'password' => 'guest'

]);

$connection->connect() or die("Cannot connect to the broker!\n");

try {

$channel = new AMQPChannel($connection);

$exchange = new AMQPExchange($channel);

$exchange->setName($exchangeName);

$exchange->setType(AMQP_EX_TYPE_FANOUT);

$exchange->declareExchange();

echo 'Send Message: ' . $exchange->publish($message) . "\n";

echo "Message Is Sent: " . $message . "\n";

} catch (AMQPConnectionException $e) {

var_dump($e);

}

// 断开连接

$connection->disconnect();

receive_logs.php

/**

* 接收消息

*/

$exchangeName = 'logs';

// 建立TCP连接

$connection = new AMQPConnection([

'host' => 'localhost',

'port' => '5672',

'vhost' => '/',

'login' => 'guest',

'password' => 'guest'

]);

$connection->connect() or die("Cannot connect to the broker!\n");

$channel = new AMQPChannel($connection);

$exchange = new AMQPExchange($channel);

$exchange->setName($exchangeName);

$exchange->setType(AMQP_EX_TYPE_FANOUT);

$exchange->declareExchange();

$queue = new AMQPQueue($channel);

$queue->setFlags(AMQP_EXCLUSIVE);

$queue->declareQueue();

$queue->bind($exchangeName);

var_dump("Waiting for message...");

// 消费队列消息

while(TRUE) {

$queue->consume('processMessage');

}

// 断开连接

$connection->disconnect();

function processMessage($envelope, $queue) {

$msg = $envelope->getBody();

var_dump("Received: " . $msg);

$queue->ack($envelope->getDeliveryTag()); // 手动发送ACK应答

}

演示流程

打开两个终端,一个消费者队列负责将日志写入文件:

php receive_logs.php > logs_from_rabbit.log

一个负责将日志输出到屏幕:

php receive_logs.php

然后再打开一个终端,将日志信息发送到所有队列:

php emit_logs.php

这样,会发现所有队列会同时接收到日志并进行相应的处理。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值