前面我们已经学了,
我们知道通过 ActiveMQ 的一个管理后台可以查看任务队列。
用 PHP 来操作 ActiveMQ,我们可以借助一个第三方扩展。
下载:
- composer require fusesource/stomp-php:2.0.*
然后新建 test.php:
- <?php
- require __DIR__.'/vendor/autoload.php'; //引入自动加载的文件
- $connect = new \FuseSource\Stomp\Stomp('tcp://10.211.55.13/:61613');
- $connect->connect();
- $userId = 1001;
- $result = $connect->send('email',$userId); //比如发邮件
- var_dump($result);
发送消息成功,打印
- bool(true)
我们在 ActiveMQ 自带的管理后台查看,确实有一个名为 "email" 的队列。
上面我们发送的一个 id,我们还可以发送 json 数据。
- $data = array('id'=>1001,'email'=>,'content'=>'test');
- $result = $connect->send('email',json_encode($data));
我们在 MQ 后台可以查看消息详细
上面的代码到这里,还不够完美。如果我们服务器重启了 activemq,没有处理的消息会丢失。
这个时候我们需要用到
方法的第三个参数。
- send()
- //消息持久化 persistent为true,字符串的'true'
- $result = $connect->send('email',json_encode($data),array('persistent'=>'true'));
给 mq 服务器发送消息 (email 消息)。
那么在 mq 的队列中的任务,又是怎么处理的呢?
- <?php
- require __DIR__.'/vendor/autoload.php'; //引入自动加载的文件
- $connect = new \FuseSource\Stomp\Stomp('tcp://10.211.55.13/:61613');
- $connect->connect();
- //订阅队列消息
- $connect->subscribe('email');
- if ($connect->hasFrameToRead()){
- $frame = $connect->readFrame();
- print_r($frame);
- }
在 mq 服务端,订阅 (监听) 队列消息。
在服务端是命令行下执行:
如果有没有处理的消息,可以读取出来,打印结果如下:
- php mqServer.php
- FuseSource\Stomp\Frame Object
- (
- [command] => MESSAGE
- [headers] => Array
- (
- [expires] => 0
- [destination] => /queue/email
- [priority] => 4
- [message-id] => ID:localhost.localdomain-38488-1488196907415-3:2:-1:1:1
- [timestamp] => 1489477647931
- )
- [body] => {"id":1001,"email":"","content":"test"}
- )
就把我们发送的内容读取出来了。
- body
我们循环读取 (死循环) 一直等待新消息:
- do{
- if ($connect->hasFrameToRead()){
- $frame = $connect->readFrame();
- print_r($frame->body);
- }
- } while (true);
处理消息之后(在发送邮件等业务完成之后),要通知 mq 我处理了该条消息了
- if ($connect->hasFrameToRead()){
- $frame = $connect->readFrame();
- //print_r($frame->body);
- //做业务逻辑
- //具体发送邮件的业务
- //send email
- //最后通知mq,我们已经处理了该条消息
- $connect->ack($frame);
- }
我们还可以在优化一下代码,解决死循环,控制循环 (这里是一种方案演示)
- do{
- //会等待,直到有可用消息,才执行后面代码
- if ($connect->hasFrameToRead()){
- $frame = $connect->readFrame();
- //print_r($frame->body);
- //做业务逻辑
- //具体发送邮件的业务
- //send email
- sleep(2); //模拟延时
- //最后通知mq,我们已经处理了该条消息
- $connect->ack($frame);
- }
- //控制循环
- $next = true;
- if (file_exists(__DIR__.'/stop')){
- //如果有名为stop的一个文件
- //就不循环了
- $next = false;
- }
- } while ($next);
来源: http://www.bubuko.com/infodetail-1982913.html