PHP实现异步调用方法研究与分享


Posted in PHP onOctober 27, 2011

这样就出现了一个问题,一个客户端的相应服务端可能执行1秒也有可能执行1分钟,这样浏览器就会一直处于等待状态,如果程序执行缓慢,用户可能就没耐心关掉了浏览器。

而有的时候我们不需要关心程序执行的结果,没有必要这样浪费时间和耐心等待,那我们就要想出办法让程序不收等待在后台静默执行。

比如现在有一个场景,给1000个用户发送一封推荐邮件,用户输入或者导入邮件账号了提交服务器执行发送。

<?php 
$count=count($emailarr); 
for($i=0;$i<$count;$i++) 
{ 
sendmail(.....);//发送邮件 
} 
?>

这段代码用户体验极差,也无法实际运用,首先发送这么多邮件会产生服务器运行超时,其实漫长的用户等待时间会让用户对系统产品怀疑和失去信心。但是用户不需要等待到1000封邮件都发送完毕了才提交发送成功,我们完全可以提交后台后直接给用户提示发送成功,然后让后台程序静默依次发送。
这个时候我们就需要“异步执行”技术来执行代码,异步执行的特点是后台静默执行,用户无需等待代码的执行结果,使用异步执行的好处:
1.摆脱了应用程序对单个任务的依赖性
2.提高了程序的执行效率
3.提高了程序的扩展性
4.在一定场景提高了用户体验
5.因为PHP不支持多线程,使用异步调用的请求多个HTTP的方式达到了程序并行执行效果,但是注意的是请求的HTTP过多的话,会大大加大了系统的开销
PHP异步执行的常用方式:
1.客户端页面采用AJAX技术请求服务器
1. 最简单的办法,就是在返回给客户端的HTML代码中,嵌入AJAX调用,或者,嵌入一个img标签,src指向要执行的耗时脚本。
这种方法最简单,也最快。服务器端不用做任何的调用。
但是缺点是,一般来说Ajax都应该在onLoad以后触发,也就是说,用户点开页面后,就关闭,那就不会触发我们的后台脚本了。
而使用img标签的话,这种方式不能称为严格意义上的异步执行。用户浏览器会长时间等待php脚本的执行完成,也就是用户浏览器的状态栏一直显示还在load。
当然,还可以使用其他的类似原理的方法,比如script标签等等

2.popen()函数
resource popen ( string command, string mode );
//打开一个指向进程的管道,该进程由派生给定的 command 命令执行而产生。打开一个指向进程的管道,该进程由派生给定的 command 命令执行而产生。
所以可以通过调用它,但忽略它的输出。
pclose(popen("/home/xinchen/backend.php &", 'r'));

这个方法避免了第一个方法的缺点,并且也很快。但是问题是,这种方法不能通过HTTP协议请求另外的一个WebService,只能执行本地的脚本文件。并且只能单向打开,无法穿大量参数给被调用脚本。
并且如果,访问量很高的时候,会产生大量的进程。如果使用到了外部资源,还要自己考虑竞争。

3.CURL扩展
CURL是一个强大的HTTP命令行工具,可以模拟POST/GET等HTTP请求,然后得到和提取数据,显示在"标准输出"(stdout)上面

$ch = curl_init(); 
$curl_opt = array(CURLOPT_URL, 'http://www.example.com/backend.php', 
CURLOPT_RETURNTRANSFER, 1, 
CURLOPT_TIMEOUT, 1,); 
curl_setopt_array($ch, $curl_opt); 
curl_exec($ch); 
curl_close($ch);

使用CURL需要设置CUROPT_TIMEOUT为1(最小为1,郁闷)。也就是说,客户端至少必须等待1秒钟。
4.fscokopen()函数
fsockopen是一个非常强大的函数,支持socket编程,可以使用fsockopen实现邮件发送等socket程序等等,使用fcockopen需要自己手动拼接出header部分
官方文档: http://cn.php.net/fsockopen/
$fp = fsockopen("www.example.com", 80, $errno, $errstr, 30); 
if (!$fp) { 
echo "$errstr ($errno)<br />\n"; 
} else { 
$out = "GET /backend.php / HTTP/1.1\r\n"; 
$out .= "Host: www.example.com\r\n"; 
$out .= "Connection: Close\r\n\r\n"; fwrite($fp, $out); 
/*忽略执行结果 
while (!feof($fp)) { 
echo fgets($fp, 128); 
}*/ 
fclose($fp); 
}

所以,总体来看,最好用,最简单的还是第一种方法。
最完美的应该是最后一种,但是比较复杂
如果有更好的办法,欢迎交流。
PHP 相关文章推荐
swfupload 多文件上传实现代码
Aug 27 PHP
利用PHP扩展vld查看PHP opcode操作步骤
Mar 04 PHP
yii框架配置默认controller和action示例
Apr 30 PHP
PHP实现递归无限级分类
Oct 22 PHP
PHP入门教程之图像处理技巧分析
Sep 11 PHP
PHP接口并发测试的方法(推荐)
Dec 15 PHP
PHP面向对象学习之parent::关键字
Jan 18 PHP
php中青蛙跳台阶的问题解决方法
Oct 14 PHP
php字符串过滤strip_tags()函数用法实例分析
Jun 24 PHP
Yii框架连表查询操作示例
Sep 06 PHP
php的instanceof和判断闭包Closure操作示例
Jan 26 PHP
Yii 框架使用Forms操作详解
May 18 PHP
提示Trying to clone an uncloneable object of class Imagic的解决
Oct 27 #PHP
PHP 小心urldecode引发的SQL注入漏洞
Oct 27 #PHP
Session保存到数据库的php类分享
Oct 24 #PHP
php中批量修改文件后缀名的函数代码
Oct 23 #PHP
基于Snoopy的PHP近似完美获取网站编码的代码
Oct 23 #PHP
php中经典方法实现判断多维数组是否为空
Oct 23 #PHP
PHP禁止页面缓存的代码
Oct 23 #PHP
You might like
PHP实现的下载远程图片自定义函数分享
2015/01/28 PHP
PHP全局变量与超级全局变量区别分析
2016/04/01 PHP
php批量修改表结构实例
2017/05/24 PHP
PHP CURL使用详解
2019/03/21 PHP
silverlight线程与基于事件驱动javascript引擎(实现轨迹回放功能)
2011/08/09 Javascript
js 判断checkbox是否选中的操作方法
2012/11/09 Javascript
jQuery获取Select选择的Text和Value(详细汇总)
2013/01/25 Javascript
使用jQuery内容过滤选择器选择元素实例讲解
2013/04/18 Javascript
javascript工厂方式定义对象
2014/12/26 Javascript
JavaScript模板引擎用法实例
2015/07/10 Javascript
谷歌Chrome浏览器扩展程序开发小记
2016/01/06 Javascript
JS组件Bootstrap dropdown组件扩展hover事件
2016/04/17 Javascript
41个Web开发者必须收藏的JavaScript实用技巧
2016/07/22 Javascript
js removeChild 方法深入理解
2016/08/16 Javascript
JS 获取HTML标签内的子节点的方法
2016/09/21 Javascript
谈谈jQuery之Deferred源码剖析
2016/12/19 Javascript
js实现数组去重方法及效率?Ρ? target=
2017/02/14 Javascript
jQuery实现导航回弹效果
2017/02/27 Javascript
DataTables添加额外的查询参数和删除columns等无用参数实例
2017/07/04 Javascript
ajax请求data遇到的问题分析
2018/01/18 Javascript
Mint-UI时间组件起始时间问题及时间插件使用
2018/08/20 Javascript
详解nodejs http请求相关总结
2019/03/31 NodeJs
python中执行shell的两种方法总结
2017/01/10 Python
python使用xslt提取网页数据的方法
2018/02/23 Python
让你Python到很爽的加速递归函数的装饰器
2019/05/26 Python
对python中的控制条件、循环和跳出详解
2019/06/24 Python
python requests库爬取豆瓣电视剧数据并保存到本地详解
2019/08/10 Python
python简单实现最大似然估计&amp;scipy库的使用详解
2020/04/15 Python
css3的transform中scale缩放详解
2014/12/08 HTML / CSS
Java如何支持I18N?
2016/10/31 面试题
市场营销专科应届生求职信
2013/11/24 职场文书
会计学个人自荐信模板
2013/12/13 职场文书
离婚协议书标准格式
2014/10/04 职场文书
夫妻忠诚协议书范本
2014/11/17 职场文书
青年文明号汇报材料
2014/12/23 职场文书
继续教育个人总结
2015/03/03 职场文书