PHP中使用ActiveMQ实现消息队列

PHP中使用ActiveMQ实现消息队列

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

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

composer require fusesource/stomp-php:2.0.*
1
然后新建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'=>'[email protected]','content'=>'test');
$result = $connect->send('email',json_encode($data)); 
1
2
我们在MQ后台可以查看消息详细 
这里写图片描述

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

//消息持久化 persistent为true,字符串的'true'
$result = $connect->send('email',json_encode($data),array('persistent'=>'true'));
1
2
前面我们完成了『发送』
给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":"[email protected]","content":"test"}
)

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

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

do{
    if ($connect->hasFrameToRead()){
        $frame = $connect->readFrame();
        print_r($frame->body);
    }
} while (true);
1
2
3
4
5
6
处理消息之后(在发送邮件等业务完成之后),要通知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);

更多内容请关注我的订阅号:

猜你喜欢

转载自my.oschina.net/yonghan/blog/1648615