关于PHP实现异步操作的研究
1.为啥php需要异步操作?
一般来说php适用的场合是web页面展示等耗时比较短的任务,如果对于比较花时间的操作如resize图片、大数据导入、批量发送edm、sms等,就很容易出现操作超时情况。你可以说我可以设置无限超时时间,等等你也要知道php有一个工作模式是fastcgi,php无限不超时,不代表fastcgi相应不超时……如果你还想说要fastcgi相应永不超时,我建议你应该跟你们的运维人员讨论去……
这个时候异步的操作就发挥他的作用了,由于是非阻塞操作,操作会即时返回,然后在后台再慢慢干活。管你超时不超时的,我就没有在当前的进程/线程下干活。看吧是不是很美好,不过其实这也是个坑……
2.php可以实现异步操作吗?
答案是肯定的,不过网上各种的纯php实现得就有点别扭了。socket模式、挂起进程模式、有的还直接fork进程。很好,各路神仙各显神通。如果运维人员看到的话,一定会×××××你们的,不把web server跑死才怪……
那还有其他更好的方法去实现这个异步操作的可能么?有,现在我们只有想怎么开外挂了。查一下pecl主流的外挂方案有一堆的××mq(消息队列),其中有个用于任务分配的外挂进入了我们的视线gearman(其实这家伙才是角,我就不详细介绍了,点连接看介绍)。
3.为啥选择gearman?
别的不说,就说他的client多,支持很多语言的client,你可以使用大部分你喜欢的语言去写worker。我个人是很烦语言之争,你喜欢用神码语言写worker都随你喜欢。有数据持久化支持(就是把队列保存到数据库介质中,那故障恢复也好做),有群集支持(其实很多××mq都有这些功能)。pecl上有扩展,也有纯php实现扩展。反正这个gearman也活了很久了,杂七杂八的问题都基本上解决了。
4.基本思路
有了gearman这外挂就简单多了。就是向gearman发送一个任务,把执行的任务发出去,然后等待worker去调用php cli去运行我们的php代码。
我就写了一下一个python的worker(别问我为啥用python,1.我会python,2.linux下不用装runtime),你可以自己根据思路写一个php的worker,不过嘛,本人是不太信得过php跑的worker。其他语言饭可以用java、node.js 或者其他语言实现一个worker试试。对用golang写worker有兴趣的朋友可以找我。
不好意思,里面是没有注释的。一个配置文件,一个py脚本。基本的功能也就是分析一下调用的参数,然后调用php cli,就是那样子而已。要让py脚本跑起来请自行安装python的gearman模块。
然后到php的部分先上测试代码:
<?php
require_once 'phpasyncclient.php';
date_default_timezone_set('asia/shanghai');
class asynctest {
const
log_file = '/debug.log';
static public function run() {
if (phpasyncclient::in_callback(__file__)) {
self::log('php async callback');
phpasyncclient::parse();
return;
}
if (phpasyncclient::is_main(__file__)) {
self::log('main run');
$async_call = phpasyncclient::getinstance();
$async_call->asynccall('asynctest', 'callback', array(
'content' => 'hello world!!!',
), array(
'class' => 'asynctest',
'method' => 'callback',
'params' => array(
'content' => 'hello callback!',
),
), __file__);
return;
}
}
static public function callback($args) {
self::log('asynctest callback run');
self::log('asynctest callback args:'.print_r($args, true));
}
static public function log($content) {
$fullname = dirname(__file__).self::log_file;
$content = date('[y-m-d h:i:s]').$content."\n";
file_put_contents($fullname, $content, file_append);
}
}
asynctest::run();
就3个静态方法,一个是用于调试的log方法,其他都是字面意思。这个例子是对这种调用方式有个初步印象。然后直接上php的所有源码:
然后应该会有很多人会说,win下安装不了gearman……所以我把java版的gearman server也放上去吧。
5.结论
经过以上配置犀牛一样大的家伙后(要装一个gearman,还要跑个py脚本),我们基本上就使php拥有了异步调用功能,当然其中还有一个状态维护神马的要自己去实现。所以发现,其实这个方案不咋样,太复杂了。还是使用一些web service的方式去做web callback会好点(问题是web callback一样会超时……),这个请留意后续。
为防止上面的代码无法下载,特打包
原文链接:http://my.oschina.net/wakanoc/blog/101789
上一篇: 的确是个傻逼
下一篇: 没有,有动感地带要吗