轻松上手 PHP + RabbitMQ 消息发布与订阅
场景
开发一个电竞比分网系统,有许多模块依赖实时比赛状态(待开始、进行中、已结束、异常),比赛状态 进行中->已结束 由图像识别处理,识别到比赛结束后向消息队列发送某场比赛的状态信息,其他模块只需订阅队列消息获取比赛状态更新并进行对于逻辑处理
RabbitMQ 概念
消息中间件
消息(Message)是指在应用间传送的数据。消息可以非常简单,比如只包含文本字符串,JSON,也可以很复杂,比如内嵌对象。
消息队列中间件(Message Queue Middleware,MQ)是指利用高效可靠的消息传递机制进行与平台无关的数据交流,并基于数据通信来进行分布式系统的集成。
MQ典型应用场景:
异步处理流量削峰日志处理应用解耦RabbitMQ 是采用 Erlang 语言实现 AMQP (Advanced Message Queuing Protocol),高级消息队列协议)的消息中间件,它最初起源于金融系统,用于在分布式系统中存储转发消息
Producer: 生产者,
生产者创建消息,然后发布到RabbitMQ中。消息一般可以包含2个部分:消息体和标签(Label) 。消息体也可以称之为payload,在实际应用中,消息体一般是一个带有业务逻辑结构的数据,比如一个 JSON 字符串。
Consumer: 消费者
消费者连接到RabbitMQ服务器,并订阅到队列上。当消费者消费一条消息时,只是消费消息的消息体(payload)。
Broker: 消息中间件的服务节点
对于RabbitMQ来说,一个RabbitMQ Broker可以简单地看作一个RabbitMQ服务节点,或者 RabbitMQ服务实例 。
Queue: 队列
Queue是RabbitMQ 内部对象,用于存储消息。RabbitMQ中消息都只能存储在队列中,队列的特性是先进先出。
RabbitMQ 安装运行
使用Docker安装RabbitMQ$ docker pull rabbitmq:3.8.3-management 运行服务端口:5672管理端端口:15672$ docker run --name rabbitmq -d -p 5672:5672 -p 15672:15672 -v /data:/var/lib/rabbitmq rabbitmq:3.8.3-management web管理端登录查看(http://127.0.0.1:15672)默认账号:guest,默认密码:guest
rabbitmq-admin
PHP 简单使用
安装
$ composer require php-amqplib/php-amqplib发布者
<?php require_once __DIR__ . /../vendor/autoload.php; use PhpAmqpLib\Connection\AMQPStreamConnection; use PhpAmqpLib\Message\AMQPMessage; $exchange = Gaming; $connection = new AMQPStreamConnection(127.0.0.1, 5672, ar414, ar414, test); $channel = $connection->channel(); $channel->exchange_declare($exchange, direct, false, false, false); for ($i = 0; $i < 100; $i++) { $routes = [dota, csgo, lol]; $key = array_rand($routes); $arr = [ match_id => $i, status => rand(0,3) ]; $data = json_encode($arr); $msg = new AMQPMessage($data); $channel->basic_publish($msg, $exchange, $routes[$key]); echo 发送 .$routes[$key]. 消息: . $data . PHP_EOL; } $channel->close(); $connection->close();订阅者
<?php require_once __DIR__ . /../vendor/autoload.php; use PhpAmqpLib\Connection\AMQPStreamConnection; $exchange = Gaming; $routerKey = lol; //只订阅LOL消息 $connection = new AMQPStreamConnection(127.0.0.1, 5672, ar414, ar414, test); $channel = $connection->channel(); $channel->exchange_declare($exchange, direct, false, false, false); list($queueName, ,) = $channel->queue_declare("", false, false, true, false); $channel->queue_bind($queueName, $exchange, $routerKey); echo " 等待消息中..." .PHP_EOL; $callback = function ($msg) { echo 接收到消息:,$msg->delivery_info[routing_key], :, $msg->body, PHP_EOL; sleep(1); //模拟耗时执行 }; $channel->basic_consume($queueName, , false, true, false, false, $callback); while ($channel->is_consuming()) { $channel->wait(); } $channel->close(); $connection->close();运行
1. 运行某一个订阅者程序监听LOL消息队列(LolSub.php)
2. 运行发送者程序(Send.php)发送者
$ php Send.phprabbitmq-send
LOL订阅者
$ php LolSub.phplol-sub