rebitmq php,PHP实战RabbitMQ之基础使用篇

简介

RabbitMQ是一个高可用的信息中间件,学习和使用RabbitMQ非常有必要。

异步消息传递

支持各种开发语言java、python、php等

可插拔的身份验证、授权

RabbitMQ-Manager可用于管理和监视。

安装

这里直接使用docker,很方便的进行安装

拉取镜像

docker pull rabbitmq:3.8.3-management-alpine

运行

docker run --name run-rabbitmq -d -p 15672:15672 -p 5672:5672 rabbitmq

15672端口是RabbitMQ Web管理页面,直接访问:http://localhost:15672/,初始用户密码:guest

bVbHJZ0

使用

RabbitMQ作为生产者和消费者来使用时,基本上有2中场景

一个/多个生产者,多个共享消费者

一个/多个生产者,多个独立消费者

共享的消费者可以同时消费一个队列的数据,增加吞吐量

独立的消费者不共享队列,每个消费者都有自己的队列,可以定义规则从exchange中pull数据到自己的queue中

下面将通过代码来实现各种场景

基础概念

queue

数据队列,数据可以推送到queue,也可以从queue中消费

exchange 交换机

将数据推送到交换机中,队列可以绑定交换机,交换机的类型不同所支持的绑定规则也不同

fanout 没有规则,所有exchange中的数据

direct 精确匹配,只绑定routingkey指定值的数据

topic 更加灵活的规则,路由键routingkey必须是一个由.分隔开的词语,* (星号) 用来表示一个单词,# (井号) 用来表示任意数量(零个或多个)单词

封装RabbitMQ一些常用操作

namespace RabbitMQ;

use PhpAmqpLib\Connection\AMQPStreamConnection;

use PhpAmqpLib\Message\AMQPMessage;

class RabbitMQ

{

private $host = '127.0.0.1';

private $port = 5672;

private $user = 'guest';

private $password = 'guest';

protected $connection;

protected $channel;

/**

* RabbitMQ constructor.

*/

public function __construct()

{

$this->connection = new AMQPStreamConnection($this->host, $this->port, $this->user, $this->password);

$this->channel = $this->connection->channel();

}

/**

* @param $exchangeName

* @param $type

* @param $pasive

* @param $durable

* @param $autoDelete

*/

public function createExchange($exchangeName, $type, $pasive = false, $durable = false, $autoDelete = false)

{

$this->channel->exchange_declare($exchangeName, $type, $pasive, $durable, $autoDelete);

}

/**

* @param $queueName

* @param $pasive

* @param $durable

* @param $exlusive

* @param $autoDelete

*/

public function createQueue($queueName, $pasive = false, $durable = false, $exlusive = false, $autoDelete = false, $nowait = false, $arguments = [])

{

$this->channel->queue_declare($queueName, $pasive, $durable, $exlusive, $autoDelete, $nowait, $arguments);

}

/**

* 生成信息

* @param $message

*/

public function sendMessage($message, $routeKey, $exchange = '', $properties = [])

{

$data = new AMQPMessage(

$message, $properties

);

$this->channel->basic_publish($data, $exchange, $routeKey);

}

/**

* 消费消息

* @param $queueName

* @param $callback

* @throws \ErrorException

*/

public function consumeMessage($queueName, $callback, $tag = '', $noLocal = false, $noAck = false, $exclusive = false, $noWait = false)

{

$this->channel->basic_consume($queueName, $tag, $noLocal, $noAck, $exclusive, $noWait, $callback);

while ($this->channel->is_consuming()) {

$this->channel->wait();

}

}

/**

* @throws \Exception

*/

public function __destruct()

{

$this->channel->close();

$this->connection->close();

}

}

多个共享消费者

多个消费者可以增加消费速度,提供系统吞吐量

bVbHKp8

小二,直接上代码吧

生产者代码

require_once '../../vendor/autoload.php';

use RabbitMQ\RabbitMQ;

use PhpAmqpLib\Message\AMQPMessage;

$rabbit = new RabbitMQ();

$queueName = 'test-single-queue';

$rabbit->createQueue($queueName,false,true,false,false);

for ($i = 0; $i < 10000; $i++) {

$rabbit->sendMessage($i . "this is a test message.", $queueName,'',[

'delivery_mode' => AMQPMessage::DELIVERY_MODE_PERSISTENT //消息持久化,重启rabbitmq,消息不会丢失

]);

}

unset($rabbit);//关闭连接

运行生产者php Producer,在manager web页面可以可看到这个queue信息

bVbHKUF

消费者代码

require_once '../../vendor/autoload.php';

use RabbitMQ\RabbitMQ;

$rabbit = new RabbitMQ();

$queueName = 'test-single-queue';

$callback = function ($message){

var_dump("Received Message : " . $message->body);//print message

sleep(2);//处理耗时任务

$message->delivery_info['channel']->basic_ack($message->delivery_info['delivery_tag']);//ack

};

$rabbit->consumeMessage($queueName,$callback);

unset($rabbit);//关闭连接

运行消费者二次php Consumer.php

bVbHKT5

bVbHKT6

可以看到二个消费者不会重复消费message

也可通过manager web看到此queue的message正在被消费

bVbHKUH

多个独立消费者

RabbitMQ生产者将message推送到exchange,通过将多个queue与exchange进行绑定,来实现多个独立消费者

定义一个topic类型的交换机,消费规则是:test.ex.加一个单词

require_once '../../vendor/autoload.php';

use RabbitMQ\RabbitMQ;

$rabbit = new RabbitMQ();

$exchangeName = 'test-ex-topic';

$queueName = 'test-consumer-ex-topic';

$routingKey = 'test.ex.*';//消费规则定义

//创建队列

$rabbit->createQueue($queueName, false, true);

//绑定到交换机

$rabbit->bindQueue($queueName, $exchangeName, $routingKey);

//消费

$callback = function ($message) {

var_dump("Received Message : " . $message->body);//print message

sleep(2);//处理耗时任务

$message->delivery_info['channel']->basic_ack($message->delivery_info['delivery_tag']);//ack

};

$rabbit->consumeMessage($queueName, $callback);

unset($rabbit);//关闭连接

启动消费者php Consumer.php

定义生产者,会向2个不同的routingkey中推送message

require_once '../../vendor/autoload.php';

use PhpAmqpLib\Exchange\AMQPExchangeType;

use PhpAmqpLib\Message\AMQPMessage;

use RabbitMQ\RabbitMQ;

$rabbit = new RabbitMQ();

$routingKey1 = 'test.ex.queue1';

$routingKey2 = 'test.ex.queue2';

$exchangeName = 'test-ex-topic';

$rabbit->createExchange($exchangeName, AMQPExchangeType::TOPIC, false, true, false);

//向交换机和routingkey = test-ex-queue1中推送10000条数据

for ($i = 0; $i < 10000; $i++) {

$rabbit->sendMessage($i . "this is a queue1 message.", $routingKey1, $exchangeName, [

'delivery_mode' => AMQPMessage::DELIVERY_MODE_PERSISTENT //消息持久化,重启rabbitmq,消息不会丢失

]);

}

//向交换机和routingkey = test-ex-queue2中推送10000条数据

for ($i = 0; $i < 10000; $i++) {

$rabbit->sendMessage($i . "this is a queue2 message.", $routingKey2, $exchangeName, [

'delivery_mode' => AMQPMessage::DELIVERY_MODE_PERSISTENT //消息持久化,重启rabbitmq,消息不会丢失

]);

}

unset($rabbit);//关闭连接

运行生产者php Producer.php,可以看到消费者有2万条message可以消费,包含了2个routingkey中的数据

bVbHK4U

代码

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值