项目中常常会有后台运行任务的需求,好比发送邮件时,由于要链接邮件服务器,每每须要5-10秒甚至更长时间,若是能先给用户一个成功的提示信息,而后在后台慢慢处理发送邮件的操做,显然会有更好的用户体验。javascript
为了实现相似的需求,Web项目中通常的实现方法是使用消息队列(Message Queue),好比MemcacheQ,RabbitMQ等等,都是很著名的产品。php
消息队列说白了就是一个最简单的先进先出队列,队列的一个成员就是一段文本。正是由于消息队列实在太简单了,当拿着消息队列时,反而有点无从下手的感受,由于这仅仅一个发送邮件的任务,就会引伸出不少问题:java
对以上这些问题,目前为止我能找到的最好答案,并非来自php,而是来自Ruby的项目Resque,正是因为Resque清晰简单的解决了后台任务带来的一系列问题,Resque的设计也被Clone到Python、php、NodeJs等语言:好比Python下的pyres以及PHP下的php-resque等等,这里有各类语言版本的Resque实现,而在本篇日志里,咱们固然要以PHP版本为例来讲明如何用php-resque运行一个后台任务,可能一些细节方面会与Ruby版有出入,可是本文中以php版为准。python
Resque是这样解决这些问题的:git
其实从上面的问题已经能够看出,只靠一个消息队列是没法解决全部问题的,须要新的角色介入。在Resque中,一个后台任务被抽象为由三种角色共同完成:github
那么基于这个划分,一个后台任务在Resque下的基本流程是这样的:web
至此就能够完整的运行完一个后台任务。redis
在Resque中,还有一个很重要的设计:一个Worker,能够处理一个队列,也能够处理不少个队列,而且能够经过增长Worker的进程/线程数来加快队列的执行速度。服务器
须要提早说明的是,因为涉及到进程的开辟与管理,php-resque使用了php的PCNTL函数,因此只能在Linux下运行,而且须要php编译PCNTL函数。若是但愿用Windows作一样的工做,那么能够去找找Resque的其余语言版本,php在Windows下很是不适合作后台任务。composer
以Ubuntu12.04LTS为例,Ubuntu用apt安装的php已经默认编译了PCNTL函数,无需任何配置,如下指令均为root账号
apt-get install redis-server
apt-get install curl
cd /usr/local/bin
curl -s http://getcomposer.org/installer | php chmod a+x composer.phar alias composer='/usr/local/bin/composer.phar'
假设web目录在/opt/htdocs
apt-get install git git-core
cd /opt/htdocs
git clone git://github.com/chrisboulton/php-resque.git cd php-resque composer install
其实php-resque已经给出了简单的例子, demo/job.php文件就是一个最简单的Job:
class PHP_Job { public function perform() { sleep(120); fwrite(STDOUT, 'Hello!'); } }
这个Job就是在120秒后向STDOUT输出字符Hello!
在Resque的设计中,一个Job必须存在一个perform方法,Worker则会自动运行这个方法。
php-resque也给出了最简单的插入队列实现 demo/queue.php:
if(empty($argv[1])) { die('Specify the name of a job to add. e.g, php queue.php PHP_Job'); } require __DIR__ . '/init.php'; date_default_timezone_set('GMT'); Resque::setBackend('127.0.0.1:6379'); $args = array( 'time' => time(), 'array' => array( 'test' => 'test', ), ); $jobId = Resque::enqueue('default', $argv[1], $args, true); echo "Queued job ".$jobId."\n\n";
在这个例子中,queue.php须要以cli方式运行,将cli接收到的第一个参数做为Job名称,插入名为'default'的队列,同时向屏幕输出刚才插入队列的Job Id。在终端输入:
php demo/queue.php PHP_Job
结果能够看到屏幕上输出:
Queued job b1f01038e5e833d24b46271a0e31f6d6
即Job已经添加成功。注意这里的Job名称与咱们编写的Job Class名称保持一致:PHP_Job
php-resque一样提供了查看Job运行状态的例子,直接运行:
php demo/check_status.php b1f01038e5e833d24b46271a0e31f6d6
能够看到输出为:
Tracking status of b1f01038e5e833d24b46271a0e31f6d6. Press [break] to stop. Status of b1f01038e5e833d24b46271a0e31f6d6 is: 1
咱们刚才建立的Job状态为1。在Resque中,一个Job有如下4种状态:
由于没有Worker运行,因此刚才建立的Job仍是等待状态。
此次咱们直接编写demo/resque.php:
<?php date_default_timezone_set('GMT'); require 'job.php'; require '../bin/resque';
能够看到一个Worker至少须要两部分:
在终端中运行:
QUEUE=default php demo/resque.php
前面的QUEUE部分是设置环境变量,咱们指定当前的Worker只负责处理default队列。也可使用
QUEUE=* php demo/resque.php
来处理全部队列。
运行后输出为
#!/usr/bin/env php *** Starting worker
用ps指令检查一下:
ps aux | grep resque
能够看到有一个php的守护进程已经在运行了
1000 4607 0.0 0.1 74816 11612 pts/3 S+ 14:52 0:00 php demo/resque.php
再使用以前的检查Job指令
php demo/check_status.php b1f01038e5e833d24b46271a0e31f6d6
2分钟后能够看到
Status of b1f01038e5e833d24b46271a0e31f6d6 is: 4
任务已经运行完毕,同时屏幕上应该能够看到输出的Hello!
至此咱们已经成功的完成了一个最简单的Resque实例的所有演示,更复杂的状况以及遗留的问题会在下一次的日志中说明。