php 后台怎么开一个进程监听Redis的队列消息呢?用while

Posted

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了php 后台怎么开一个进程监听Redis的队列消息呢?用while相关的知识,希望对你有一定的参考价值。

redis的subscribe用pconnect链接,执行这个脚本的进程会自动监听所订阅的频道发送的消息

ini_set(‘default_socket_timeout’, -1);

$redis = new \Redis();
$redis->pconnect('127.0.0.1', 6379);

//订阅
$redis->subscribe(['msg'], 'callfun');

function callfun($redis, $channel, $msg)

var_dump([
'redis' => $redis,
'channel' => $channel,
'msg' => $msg
]);
参考技术A 建议用php作为守护进程,先fork多个子进程(worker),然后主进程定时读取redis队列,将消息分配给子进程处理。
不过这些都有一些开源的包,比如php-resque,具体实现可以看它的源码~本回答被提问者采纳

php 守护进程 (简单)

首先需要解释的是什么是守护进程。

守护进程就是在后台一直运行的进程。比如我们启动的httpd,mysqld等进程都是常驻内存内运行的程序。

针对需求进行分析:

需求:有一个常驻队列messageQueue(假设在redis内存中),这个队列会有可能有请求不定期的往队列中增加元素。同时我们要求在队列中有元素的时候,按照队列顺序将元素pop出来,并进行处理(假设这个处理只是echo ‘test’);

解决方法:

现在假设已经有了两个函数

function oPopMessageQueue(){ …} //获取队列最后一个元素;

function vDealElement($element) { …} 处理元素;

要求写出一个守护程序,完成上面的需求。

程序:

好了,这个程序很容易想到,可以使用while循环来做

while(true)
{
    if( $element  = oPopMessageQueue())
    {
        vDealElement($element);
    }   
}

考虑1 : 这个程序如果一直跑的话已经可以满足上面的需求了.

但是考虑到:1 用php进程跑有可能会由于各种情况(比如运行时间过长),进程挂了,这样程序就无法自动重连了.

方法:使用cron

我们在定时脚本中每10分钟起一个进程跑这个程序。

然后设置这个程序的运行时间为10分钟,10分钟后自动取消,于是代码变成

while(true)
{
    if($element = oPopMessageQueue())
    {
        vCheckTimeLimit();
        vDealElement($elemnt);
    }
}
 
$timeStart = 0;
function vCheckTimeLimit()
{
    global $timeStart;
    if(empty($timeStart))
    {
        $timeStart = time();
    }   
     
    if(time() - $timeStart > 60*10)
    {
        exit;
    }
}
考虑2,可能会有这种需求: 需要有随时让脚本暂停的功能:
于是考虑使用文件来增加暂停功能
 
while(true)
{
    if($element = oPopMessageQueue())
    {
        vCheckTimeLimit();
        vCheckEnd();
        vDealElement($elemnt);
    }
}
 
 
function vCheckEnd()
{
    if(file_exists("/home/JesephYe/end"))
    {
        exit;
    }
}

考虑3, 是否可以改成多线程的程序,让运行的效率更高?

这个只要把cron的10分钟起一个进程的限制改成每1分钟起一个进程就好了

这样能保证有10个线程在运行程序

但是有一个基本要求是:oPopMessageQueue()是一个原子操作

以上是关于php 后台怎么开一个进程监听Redis的队列消息呢?用while的主要内容,如果未能解决你的问题,请参考以下文章

spring-data-redis 怎么监听消息队列有消息来了

redis怎么做消息队列

消息队列为啥用redis实现

redis消息队列有没有

PHP进程间通信监控消息队列

laravel5.6 基于redis,使用消息队列(邮件推送)