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; } }
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的主要内容,如果未能解决你的问题,请参考以下文章