在laravel框架中使用mq
本文写于2018-11-28
1、部署laravel项目
https://github.com/laravel/laravel 通过git克隆项目,或者下载zip包然后解压等方式都可以把laravel框架源码下载下来。
然后composer install 安装各种依赖
然后复制.env.example 为.env文件,执行php artisan key:generate 生成APP_KEY
2、上传文件到github【备注:这一步可以略过】
1)git bash 到项目目录,然后git init,初始化本地仓库
2)git remote add origin 远程库地址
3)git add . 把文件缓存到缓冲区
4)git commit -m '初始提交'
5)git push origin master 推送本地代码到远程库master分支
3、安装mq依赖
从 https://packagist.org/?query=rabbitmq 选择一个依赖包

从上面截图看到,能用的就是第一个和最后一个依赖。暂时先使用第一个依赖,后续有时间补充下第二个依赖的使用。
composer require php-amqplib/php-amqplib 即可安装依赖
4、laravel框架中使用mq
1)在config目录下新增mq.php,文件内容:
<?php
return [
'host' => env('MQ_HOST', '127.0.0.1'),
'port' => env('MQ_PORT', 5672),
'user' => env('MQ_USER', 'guest'),
'password' => env('MQ_PASSWORD', 'guest'),
'queue' => env('MQ_QUEUE', 'default'),
'exchange' => env('MQ_EXCHANGE', 'default'),
'key' => env('MQ_KEY', 'default'),
];
具体的配置信息,可以在.env文件中配置,也可以修改这个文件。
2)新增MqSend.php文件,路径:app/Console/Commands
<?php
namespace App\Console\Commands;
use Illuminate\Console\Command;
use PhpAmqpLib\Connection\AMQPConnection;
use PhpAmqpLib\Message\AMQPMessage;
class MqSend extends Command
{
/**
* 控制台命令 signature 的名称。
*
* @var string
*/
protected $signature = 'mq:send {msg}';
/**
* 控制台命令说明。
*
* @var string
*/
protected $description = 'send messages to rabbitMQ';
/**
* 创建一个新的命令实例。
*
* @return void
*/
public function __construct()
{
parent::__construct();
}
/**
* 执行控制台命令。
*
* @return mixed
*/
public function handle()
{
$host = config('mq.host');
$port = config('mq.port');
$user = config('mq.user');
$password = config('mq.password');
$queue = config('mq.queue'); // 队列名称
$exchange = config('mq.exchange'); // 交换机名称
$key = config('mq.key'); // 队列绑定交换机时配置的routingKey
$connection = new AMQPConnection($host, $port, $user, $password);
$channel = $connection->channel();
/**
* 如果管理后台上已经配置了交换机、队列,以及绑定了关系,则不需要下面的3条语句
*/
$channel->exchange_declare($exchange, 'direct', false, true, false); // 初始化交换机
$channel->queue_declare($queue, false, true, false, false); // 初始化队列
$channel->queue_bind($queue, $exchange, $key); // 将队列与某个交换机进行绑定,并使用路由关键字
$msg = '[' . date('Y-m-d H:i:s') . '] ' . $this->argument('msg');
$data = new AMQPMessage($msg, ['delivery_mode' => AMQPMessage::DELIVERY_MODE_PERSISTENT]);
$channel->basic_publish($data, $exchange, $queue);
echo "[X] Sent: $msg \n";
$channel->close();
$connection->close();
}
}
3)新增MqReceive.php文件, 路径:app/Console/Commands
<?php
namespace App\Console\Commands;
use Illuminate\Console\Command;
use PhpAmqpLib\Connection\AMQPConnection;
use PhpAmqpLib\Message\AMQPMessage;
class MqReceive extends Command
{
/**
* 控制台命令 signature 的名称。
*
* @var string
*/
protected $signature = 'mq:receive';
/**
* 控制台命令说明。
*
* @var string
*/
protected $description = 'receive messages to rabbitMQ';
/**
* 创建一个新的命令实例。
*
* @return void
*/
public function __construct()
{
parent::__construct();
}
/**
* 执行控制台命令。
*
* @return mixed
*/
public function handle()
{
$host = config('mq.host');
$port = config('mq.port');
$user = config('mq.user');
$password = config('mq.password');
$queue = config('mq.queue'); // 队列名称
$exchange = config('mq.exchange'); // 交换机名称
$key = config('mq.key'); // 队列绑定交换机时配置的routingKey
$connection = new AMQPConnection($host, $port, $user, $password);
$channel = $connection->channel();
/**
* 如果管理后台上已经配置了交换机、队列,以及绑定了关系,则不需要下面的3条语句
*/
$channel->queue_declare($queue, false, true, false, false);
echo ' [*] Waiting for messages. To exit press CTRL+C', "\n";
$callback = function($msg) {
echo " [x] Received ", $msg->body, "\n";
};
$channel->basic_consume($queue, '', false, true, false, false, $callback);
while(count($channel->callbacks)) {
$channel->wait();
}
$channel->close();
$connection->close();
}
}
4)修改app/console/Kernel.php文件,在$commands数组中增加:
protected $commands = [
Commands\MqSend::class,
Commands\MqReceive::class,
];
这样子就能在php artisan看到有mq的命令:

5)生产者发布消息

发布成功。代码中设置了交换机、队列名、理由关键词,这些默认值都是default,在rabbitMQ管理后台可以看到有新增了交换机、队列,队列里面也有消息。
【备注】windows系统上可能执行生产者脚本会报错:

这是因为windows不支持这个SOCKET_EAGAIN常量。
参考:https://github.com/php-amqplib/php-amqplib/issues/619

要改下vendor/php-amqplib/php-amqplib/PhpAmqpLib/Wire/IO/StreamIO.php文件:
// self::$ERRNO_EQUALS_EAGAIN = 'errno=' . SOCKET_EAGAIN; // windows不支持SOCKET_EAGAIN,所以会出现未定义的报错。在linux上SOCKET_EAGAIN是SOCKET_EWOULDBLOCK的别名 // https://github.com/php-amqplib/php-amqplib/issues/619 使用SOCKET_EWOULDBLOCK替换 self::$ERRNO_EQUALS_EAGAIN = 'errno=' . (defined('SOCKET_EAGAIN') ? SOCKET_EAGAIN : SOCKET_EWOULDBLOCK); self::$ERRNO_EQUALS_EWOULDBLOCK = 'errno=' . SOCKET_EWOULDBLOCK; self::$ERRNO_EQUALS_EINTR = 'errno=' . SOCKET_EINTR;
修改完之后就没问题了。
6)消费者接收消息

消息有2条,都接收到了。按Ctrl+C可以退出消费者,因为消费是阻塞,一直在等待接收消息。
5、参考文档:
1)laravel中新增artisan命令:https://laravel-china.org/docs/laravel/5.7/artisan/2276
2)laravel中使用 php-amqplib/php-amqplib 依赖包
https://segmentfault.com/a/1190000012308675
https://segmentfault.com/a/1190000011825148

浙公网安备 33010602011771号