您的位置 首页 php

轻松上手 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管理端登录查看(

rabbitmq-admin

  • 添加管理员
  • 命令行
 $ docker exec -it 89e8e968aebc bash
root@89e8e968aebc:/# rabbitmqctl add_user ar414 ar414 
root@89e8e968aebc:/# rabbitmqctl set_user_tags ar414 administrator 
  
  • Web管理端
  • 添加vhost

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.php   

rabbitmq-send

LOL订阅者

 $ php LolSub.php   

lol-sub

文章来源:智云一二三科技

文章标题:轻松上手 PHP + RabbitMQ 消息发布与订阅

文章地址:https://www.zhihuclub.com/77019.shtml

关于作者: 智云科技

热门文章

网站地图