一架梯子,一头程序猿,仰望星空!
RabbitMQ教程 > 内容正文

PHP RabbitMQ 主题模式(Topic模式)


PHP RabbitMQ主题模式(Topic模式),使用的交换机类型为Topic,跟路由模式(Direct)的区别就路由参数支持模糊匹配,架构如下图

RabbitMQ Topic模式

提示:无论使用RabbitMQ那种工作模式,区别就是使用的交换机(Exchange)类型和路由参数不一样。

1.前置教程

请先阅读下面章节,了解相关知识

2.定义Topic交换机

// 声明交换机
$channel->exchange_declare(
    'tizi365.topic', // 交换机名,需要唯一,不能重复
    'topic', // 交换机类型
    false,
    false, // 是否持久化
    false
);

提示: 无论是消息生产者还是消费者都需要交换机。

3.发送消息

我们将消息发送给交换机,由交换机根据路由规则投递消息到对应的队列。

<?php

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

// 创建rabbitmq连接
$connection = new AMQPStreamConnection('localhost', 5672, 'guest', 'guest');
// 创建Channel
$channel = $connection->channel();

// 声明交换机
$channel->exchange_declare(
    'tizi365.topic', // 交换机名,需要唯一,不能重复
    'topic', // 交换机类型
    false,
    false, // 是否持久化
    false
);


// 消息对象,参数是消息内容
$msg = new AMQPMessage("hello tizi365.com");

// 发送消息
// 注意第三个参数,路由参数
$channel->basic_publish(
    $msg, // 消息对象
    'tizi365.topic', // 交换机名字
    "www.tizi365.com" // 路由参数,可以根据需求,任意定义。
);

echo ' [x] Sent ', $msg->getBody(), "\n";

// 释放资源
$channel->close();
$connection->close();

4.接收消息

4.1.定义队列&绑定交换机

要想消费队列消息,需要先定义一个队列,然后将队列绑定到目标交换机上。

// 声明一个匿名队列
list($queue_name, ,) = $channel->queue_declare("", false, false, true, false);

// 队列绑定指定交换机
$channel->queue_bind(
    $queue_name, // 队列名
    'tizi365.topic', // 交换机名字
    "*.tizi365.com" // 绑定路由参数,这里使用了通配符 * (星号),可以匹配一个单词
);

说明:设置的路由参数都使用了*(星号)通配符,可以匹配一个单词,如果改成 # (井号),则可以匹配多个单词。

4.2.完整的接收消息代码

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

// 创建rabbitmq连接
$connection = new AMQPStreamConnection('localhost', 5672, 'guest', 'guest');
// 创建Channel
$channel = $connection->channel();

// 声明交换机
$channel->exchange_declare(
    'tizi365.topic', // 交换机名,需要唯一,不能重复
    'topic', // 交换机类型
    false,
    false, // 是否持久化
    false
);

// 声明一个匿名队列
list($queue_name, ,) = $channel->queue_declare("", false, false, true, false);

// 队列绑定指定交换机
$channel->queue_bind(
    $queue_name, // 队列名
    'tizi365.topic', // 交换机名字
    "*.tizi365.com" // 绑定路由参数,这里使用了通配符 * (星号),可以匹配一个单词
);

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

// 定义消息处理函数(这里使用匿名函数)
$callback = function ($msg) {
    // 消息处理逻辑
    echo ' [x] ', $msg->body, "\n";
};

// 创建消费者
$channel->basic_consume(
    $queue_name, // 队列名,需要消费的队列名
    '', // 消费者名,忽略,则自动生成一个唯一ID
    false,
    true, // 是否自动提交消息,即自动告诉rabbitmq消息已经处理成功。
    false,
    false,
    $callback // 消息处理函数
);

// 如果信道没有关闭,则一直阻塞进程,避免进程退出
while ($channel->is_open()) {
    $channel->wait();
}

// 释放资源
$channel->close();
$connection->close();

因为绑定交换机的时候设置的路由参数为*.tizi365.com, 匹配消息的路由参数(www.tizi365.com), 所以可以收到消息。


推荐教程