关于PHP实现异步操作的研究


Posted in PHP onFebruary 03, 2013

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有兴趣的朋友可以找我。

phpasync_worker_py

不好意思,里面是没有注释的。一个配置文件,一个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的所有源码:

php_async.zip

然后应该会有很多人会说,win下安装不了gearman……所以我把java版的gearman server也放上去吧。

java-gearman-service-0.6.6.zip

5.结论

经过以上配置犀牛一样大的家伙后(要装一个Gearman,还要跑个Py脚本),我们基本上就使PHP拥有了异步调用功能,当然其中还有一个状态维护神马的要自己去实现。所以发现,其实这个方案不咋样,太复杂了。还是使用一些web service的方式去做web callback会好点(问题是web callback一样会超时……),这个请留意后续。  

为防止上面的代码无法下载,三水点靠木特打包下载

原文链接:http://my.oschina.net/wakanoc/blog/101789

PHP 相关文章推荐
php面向对象全攻略 (四)构造方法与析构方法
Sep 30 PHP
约瑟夫环问题的PHP实现 使用PHP数组内部指针操作函数
Oct 12 PHP
php生成静态文件的多种方法分享
Jul 17 PHP
php通过修改header强制图片下载的方法
Mar 24 PHP
PHP递归创建多级目录
Nov 05 PHP
PHP引用返回用法示例
May 28 PHP
PHP MySql增删改查的简单实例
Jun 21 PHP
PHP实现的redis主从数据库状态检测功能示例
Jul 20 PHP
PHP SFTP实现上传下载功能
Jul 26 PHP
php实现的rc4加密解密类定义与用法示例
Aug 16 PHP
PHP生成短网址的思路以及实现方法的详解
Mar 25 PHP
PHP7 安装event扩展的实现方法
Oct 08 PHP
PHP数组循环操作详细介绍 附实例代码
Feb 03 #PHP
php中将字符串转为HTML的实体引用的一个类
Feb 03 #PHP
php处理文件的小例子(解压缩,删除目录)
Feb 03 #PHP
php函数array_merge用法一例(合并同类数组)
Feb 03 #PHP
php存储过程调用实例代码
Feb 03 #PHP
php中导出数据到excel时数字变为科学计数的解决方法
Feb 03 #PHP
php中删除字符串中最先出现某个字符的实现代码
Feb 03 #PHP
You might like
让的PHP代码飞起来的40条小技巧(提升php效率)
2010/04/12 PHP
php有效防止图片盗用、盗链的两种方法
2016/11/01 PHP
一个页面放2段图片滚动代码出现冲突的问题如何解决
2012/12/21 Javascript
使用jquery实现的一个图片延迟加载插件(含图片延迟加载原理)
2014/06/05 Javascript
jquery中map函数与each函数的区别实例介绍
2014/06/23 Javascript
Javascript验证上传图片大小[前台处理]
2014/07/18 Javascript
JS+CSS实现下拉列表框美化效果(3款)
2015/08/15 Javascript
js学习阶段总结(必看篇)
2016/06/16 Javascript
浅谈JS正则表达式的RegExp对象和括号的使用
2016/07/28 Javascript
微信小程序 常见问题总结(4058,40013)及解决办法
2017/01/11 Javascript
基于JavaScript实现弹幕特效
2020/08/27 Javascript
薪资那么高的Web前端必看书单
2017/10/13 Javascript
angularJs 表格添加删除修改查询方法
2018/02/27 Javascript
vue.js element-ui tree树形控件改iview的方法
2018/03/29 Javascript
浅谈VUE-CLI脚手架热更新太慢的原因和解决方法
2018/09/28 Javascript
JS实现点餐自动选择框(案例分析)
2019/12/10 Javascript
JS FormData对象使用方法实例详解
2020/02/12 Javascript
你准备好迎接vue3.0了吗
2020/04/28 Javascript
详解微信小程序「渲染层网络层错误」的解决方法
2021/01/06 Javascript
python通过colorama模块在控制台输出彩色文字的方法
2015/03/19 Python
python简单获取本机计算机名和IP地址的方法
2015/06/03 Python
Python编程使用tkinter模块实现计算器软件完整代码示例
2017/11/29 Python
python实现nao机器人身体躯干和腿部动作操作
2019/04/29 Python
Python-Seaborn热图绘制的实现方法
2019/07/15 Python
django组合搜索实现过程详解(附代码)
2019/08/06 Python
浅谈Python线程的同步互斥与死锁
2020/03/22 Python
Django {{ MEDIA_URL }}无法显示图片的解决方式
2020/04/07 Python
全面总结使用CSS实现水平垂直居中效果的方法
2016/03/10 HTML / CSS
英国剑桥包官网:The Cambridge Satchel Company
2016/08/01 全球购物
美国一家著名的儿童鞋制造商:Stride Rite
2017/01/02 全球购物
意大利时尚奢侈品店:D’Aniello Boutique
2021/01/19 全球购物
2014年民主评议党员个人总结
2014/09/24 职场文书
群众路线党员自我评议范文2014
2014/09/24 职场文书
社区公民道德宣传日活动总结
2015/03/23 职场文书
大学生学生会工作总结2015
2015/05/26 职场文书
CSS Transition通过改变Height实现展开收起元素
2021/08/07 HTML / CSS