当前位置:Gxlcms > PHP教程 > PHP使用ActiveMQ实现消息队列的方法详解

PHP使用ActiveMQ实现消息队列的方法详解

时间:2021-07-01 10:21:17 帮助过:119人阅读

本文实例讲述了PHP使用ActiveMQ实现消息队列的方法。分享给大家供大家参考,具体如下:

前面我们已经学了如何部署ActiveMQ,
我们知道通过ActiveMQ的一个管理后台可以查看任务队列。

今天

这里写图片描述
用PHP来操作ActiveMQ,我们可以借助一个第三方扩展。
下载:

  1. composer require fusesource/stomp-php:2.0.*

然后新建test.php:

  1. <?php
  2. require __DIR__.'/vendor/autoload.php'; //引入自动加载的文件
  3. $connect = new \FuseSource\Stomp\Stomp('tcp://10.211.55.13/:61613');
  4. $connect->connect();
  5. $userId = 1001;
  6. $result = $connect->send('email',$userId); //比如发邮件
  7. var_dump($result);

这里写图片描述
发送消息成功,打印bool(true)

我们在ActiveMQ自带的管理后台查看,确实有一个名为”email”的队列。
这里写图片描述

上面我们发送的一个id,我们还可以发送json数据。

  1. $data = array('id'=>1001,'email'=>'110@qq.com','content'=>'test');
  2. $result = $connect->send('email',json_encode($data));

我们在MQ后台可以查看消息详细
这里写图片描述

上面的代码到这里,还不够完美。如果我们服务器重启了activemq,没有处理的消息会丢失。
这个时候我们需要用到send()方法的第三个参数。

  1. //消息持久化 persistent为true,字符串的'true'
  2. $result = $connect->send('email',json_encode($data),array('persistent'=>'true'));

前面我们完成了『发送』

给mq服务器发送消息(email消息)。
那么在mq的队列中的任务,又是怎么处理的呢?

  1. <?php
  2. require __DIR__.'/vendor/autoload.php'; //引入自动加载的文件
  3. $connect = new \FuseSource\Stomp\Stomp('tcp://10.211.55.13/:61613');
  4. $connect->connect();
  5. //订阅队列消息
  6. $connect->subscribe('email');
  7. if ($connect->hasFrameToRead()){
  8. $frame = $connect->readFrame();
  9. print_r($frame);
  10. }

在mq服务端,订阅(监听)队列消息。
在服务端是命令行下执行:php mqServer.php
如果有没有处理的消息,可以读取出来,打印结果如下:

  1. FuseSource\Stomp\Frame Object
  2. (
  3. [command] => MESSAGE
  4. [headers] => Array
  5. (
  6. [expires] => 0
  7. [destination] => /queue/email
  8. [priority] => 4
  9. [message-id] => ID:localhost.localdomain-38488-1488196907415-3:2:-1:1:1
  10. [timestamp] => 1489477647931
  11. )
  12. [body] => {"id":1001,"email":"110@qq.com","content":"test"}
  13. )

body就把我们发送的内容读取出来了。

我们循环读取(死循环)一直等待新消息:

  1. do{
  2. if ($connect->hasFrameToRead()){
  3. $frame = $connect->readFrame();
  4. print_r($frame->body);
  5. }
  6. } while (true);

处理消息之后(在发送邮件等业务完成之后),要通知mq我处理了该条消息了

  1. if ($connect->hasFrameToRead()){
  2. $frame = $connect->readFrame();
  3. //print_r($frame->body);
  4. //做业务逻辑
  5. //具体发送邮件的业务
  6. //send email
  7. //最后通知mq,我们已经处理了该条消息
  8. $connect->ack($frame);
  9. }

我们还可以在优化一下代码,解决死循环,控制循环(这里是一种方案演示)

  1. do{
  2. //会等待,直到有可用消息,才执行后面代码
  3. if ($connect->hasFrameToRead()){
  4. $frame = $connect->readFrame();
  5. //print_r($frame->body);
  6. //做业务逻辑
  7. //具体发送邮件的业务
  8. //send email
  9. sleep(2); //模拟延时
  10. //最后通知mq,我们已经处理了该条消息
  11. $connect->ack($frame);
  12. }
  13. //控制循环
  14. $next = true;
  15. if (file_exists(__DIR__.'/stop')){
  16. //如果有名为stop的一个文件
  17. //就不循环了
  18. $next = false;
  19. }
  20. } while ($next);

更多关于PHP相关内容感兴趣的读者可查看本站专题:《PHP数据结构与算法教程》、《php程序设计算法总结》、《php字符串(string)用法总结》、《PHP数组(Array)操作技巧大全》、《PHP常用遍历算法与技巧总结》及《PHP数学运算技巧总结》

希望本文所述对大家PHP程序设计有所帮助。

人气教程排行