zoukankan      html  css  js  c++  java
  • PHP中使用ActiveMQ实现消息队列

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

    今天

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

    composer require fusesource/stomp-php:2.0.*

    然后新建test.php:

    <?php
    
    require __DIR__.'/vendor/autoload.php'; //引入自动加载的文件
    
    $connect = new FuseSourceStompStomp('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'=>'110@qq.com','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 FuseSourceStompStomp('tcp://10.211.55.13/:61613');
    $connect->connect();
    
    //订阅队列消息
    $connect->subscribe('email');
    
    if ($connect->hasFrameToRead()){
        $frame = $connect->readFrame();
        print_r($frame);
    }

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

    FuseSourceStompFrame 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":"110@qq.com","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);
  • 相关阅读:
    自定义Android Toasts风格
    用TAPI实现来电拒接
    压缩与解压缩的实现
    Android之个性化ListView实现
    如何在windows7 Home家庭版中创建Administrator管理员账户
    Android中Intent传递对象的两种方法(Serializable,Parcelable)
    Android模拟器入门
    上海西门子培训第六天(周五)
    上海西门子培训第七天(周六)
    计算机英语学习方法[摘录]
  • 原文地址:https://www.cnblogs.com/saonian/p/8384408.html
Copyright © 2011-2022 走看看