一、前言
公司项目里面用到了推送等第三方库,为了更快速的响应,决定将推送功能放入消息队列中处理。那么,如何做消息队列呢?通过网上资料,找到以下几种方案:
1)PHP+redis自己做消息队列
2)PHP-Resque
3)MemcacheQ
4)RabbitMQ
最后我们选了第二种方案,因为PHP-Resque是现成的框架,比第一种方案要方便,第二是由于我们的消息队列服务要求比较轻量级,PHP-Resque刚好满足我们的要求。
那么,开始了。
二、环境搭建
有关PHP-resque的原理可以参考文章:http://avnpc.com/pages/run-background-task-by-php-resque
1)PHP-Resque必须运行在Linux环境下,所以先找一台Linux服务器吧
2)安装PHP的Redis扩展:安装方式网上自行搜索,安装完之后用phpinfo查看Redis扩展是否安装成功
3)需要支持PCNTL函数,可以写个脚本:
echo pcntl_fork();
检验是否支持pcntl函数。若不支持,去安装。我默认安装的php环境就支持这个函数。
4)安装Redis:安装方式网上很多,这里不再赘述
三、集成
1、可以用composer安装,十分方便。
composer require chrisboulton/php-resque
2、Job、Queue、Worker
熟悉PHP-resque的原理之后,都明白PHP-resque是由三个角色组成的:Job、Queue、Worker;其中Job负责处理对应事件的逻辑,Queue用于接收队列消息,Worker常驻内存,循环POP队列中的服务。
PHP-resque提供了一个简单的demo,里面实现了Job、Queue和Worker的逻辑,完全可以参考demo的逻辑来处理你自己的逻辑。
Job类:
以上是我实现消息推送的一个Job类,Worker抛出队列服务的时候,会自动根据服务的名称去执行这个类。当然,你需要在worker里面自动加载这个类哦,这个后面再讲。
Queue类:
Queue类很简单,直接接收队列名称,Job类的名称,参数传进来就可以了。如下:
Queue::in('push', 'NotificationJob', ['push_uid' => $push_uid, 'push_msg' => $push_msg]);
这里有个很坑的地方,那就是设置Redis服务器信息的时候,我发现没有设置密码的地方,我的Redis是设置了密码的。结果只要去掉密码,程序就可以运行,加上密码,就无法将消息加入队列中,没办法,只能手动更改PHP-resque库了。
修改lib/Resque.php,增加auth方法,如下:
public static function auth($password) {
self::$password = $password;
}
修改redis方法,如下:
在
self::$redis->select(self::$redisDatabase);
之后增加如下语句:
if(self::$password) {
self::$redis->auth(self::$password);
}
ok了,现在redis有密码也可以加入消息队列了。
Worker常驻内存程序Resque.php:
直接拿demo里面的程序来稍作改造即可,第一:加入自动加载。第二:引入库文件路径要改一下,第三:设置Redis密码
程序逻辑基本就是这样,现在,让worker脚本常驻内存。
按照资料说的执行命令:QUEUE=* php Resque.php &
然后写个程序调取上面的Queue类写队列,如果Resque.php按照预期推送了消息,就说明逻辑ok了,如果没有,可以用fwrite(STDOUT, 'xxxx')来调试问题出在哪里了。
但是,用上面的命令有个问题,一按control+c,进程就退出了,这样就没办法保证worker常驻内存了。
这个不行,于是用nohup -- QUEUE=* nohup php Resque.php &
运行之后,用ps -ef|grep Resque,如果看到有相应的进程,就说明OK了
我试过之后,的确是可以的,Worker的确常驻内存了。
但是,还有另外一个问题,如果进程不小心被干死了怎么办,用kill -9 进程号杀死进程,worker是没有自动重启的,那肯定有问题,我们需要worker进程在遇到异常情况被杀死的时候能够自动重启。
于是,我们想到了shell脚本。可以参考资料:https://segmentfault.com/q/1010000006666186
我的shell脚本命名为Resque.sh,代码如下:
#!/bin/sh
PREFIX=/home/eechen
INTERVAL=1
QUEUE=* nohup php ${PREFIX}/Resque.php >>${PREFIX}/Resque.log 2>&1 & echo $! > ${PREFIX}/Resque.pid
while [ 1 ]; do
if [ ! -d /proc/`cat ${PREFIX}/Resque.pid` ]; then
QUEUE=* nohup php ${PREFIX}/Resque.php >>${PREFIX}/Resque.log 2>&1 & echo $! > ${PREFIX}/Resque.pid
echo 'NEW_PID:'`cat ${PREFIX}/Resque.pid && date '+%Y-%m-%d %H:%M:%S'`
fi
sleep ${INTERVAL}
done
脚本里面的PREFIX需要根据你自己的实际路径来修改。
然后运行:
nohup /path/to/Resque.sh >>/path/to/Resque.sh.log 2>&1 &
让脚本常驻内存
运行之后,kill -9 worker的进程号试试,kill之后再查看进程,又新建了一个进程,没有问题了。
————————————————
版权声明:本文为CSDN博主「coder麻雀」的原创文章,遵循CC 4.0 BY-SA版权协议,转载请附上原文出处链接及本声明。
原文链接:https://blog.csdn.net/maquealone/article/details/75333349