PHP 异步后台处理

PHP 作为后台的接口服务器已经很常见,在实际应用场景中经常需要异步后台处理。

推荐:《PHP视频教程》

PHP 当然具有它能作为后台服务器的优势之处,但是,在处理一些客户端并不关心的结果时,就显出它的弊端了,没有异步执行的机制。

就比如我们想做一些对于某次客户端访问php的性能记录(包括开始时间、结束时间、此次结果状态等)的记录时,客户端当然想的是php的本次处理能够早点返回,拿到结果,而如果安装常规的方案,客户端就得等php做完性能记录之后,才能拿到结果。

相当于你去银行去查你现在的余额,而柜员跑过去跟其他人闹了一会儿的磕,在来告诉你的结果一样。

所以,很多时候,就需要一种php能执行异步操作。

PHP 如何实现异步处理呢?

其中一种方案就是利用php的系统调用,开启新的进程来实现。

php 提供了fsockopen函数,此函数的功能为初始化一个套接字连接到指定主机,默认情况下将以阻塞模式开启套接字连接。

当然你可以通过stream_set_blocking()将它转换到非阻塞模式。这是关键。

所以,思路就是:开启一个非阻塞的套接字连接到本机,本机收到之后作一些耗时处理。

类似这样的处理代码(文件posttest.php):$fp = fsockopen($php_Path,80);

if (!$fp) {

LMLog::error("fsockopen:err" );

} else {

$out = "GET /album/action/album_write_friends_thread_record.php?key=&u= HTTP/1.1\r\n";

$out .= "Host: ".$php_Path."\r\n";

$out .= "Connection: Close\r\n\r\n";

stream_set_blocking($fp,true);

stream_set_timeout($fp,1);

fwrite($fp, $out);

usleep(1000);

fclose($fp);

}

这里,usleep(1000) 非常关键,它能保证这个请求能发出去。

我们在来看处理的代码逻辑(文件album_write_friends_thread_record.php):<?php

/**

* Created by PhpStorm.

* User: Administrator

* Date: 2016-09-23

* Time: 09:26

*/

/**

* 客户端调用服务器接口页面

* user: guwen

*/

sleep(20);// 睡眠20s

?>

实际上,我们服务器在执行fsockopen 那段程序时,就不会再等20s之后才能返回给客户端,

而是发出这个请求之后,即返回客户端,销毁进程,而把剩余的工作交由其他进程慢慢做去,这就实现了php的异步。

PHP 异步执行的4种常用方式

客户端与服务器端是通过HTTP协议进行连接通讯,客户端发起请求,服务器端接收到请求后执行处理,并返回处理结果。

有时服务器需要执行很耗时的操作,如处理下载、消息下发、邮件发送等,这个操作的结果并不需要返回给客户端。

但因为php是同步执行的,所以客户端需要等待服务处理完才可以进行下一步。

因此,对于耗时的操作适合异步执行,服务器接收到请求后,处理完客户端需要的数据就先返回,剩余耗时的操作再异步在服务器后台执行。

PHP异步执行的常用方式常见的有以下几种,可以根据各自优缺点进行选择:

1. ajax 请求

客户端页面采用AJAX技术请求服务器$.get("doRequest.php", { name: "fdipzone"} );

优点:最简单,也最快,就是在返回给客户端的HTML代码中,嵌入AJAX调用,或者,嵌入一个img标签,src指向要执行的耗时脚本。

缺点:一般来说Ajax都应该在onLoad以后触发,也就是说,用户点开页面后,就关闭,那就不会触发我们的后台脚本了。

而使用img标签的话,这种方式不能称为严格意义上的异步执行。用户浏览器会长时间等待php脚本的执行完成,也就是用户浏览器的状态栏一直显示还在load。

当然,还可以使用其他的类似原理的方法,比如script标签等等。

2. popen()函数

该函数打开一个指向进程的管道,该进程由派生给定的 command 命令执行而产生。

打开一个指向进程的管道,该进程由派生给定的 command 命令执行而产生。

所以可以通过调用它,但忽略它的输出。使用代码如下:// popen — 打开进程文件指针

resource popen ( string $command , string $mode )

pclose(popen('php /home/fdipzone/doRequest.php &', 'r'));

优点:避免了第一个方法的缺点,并且执行速度快。

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

1)只能在本机执行

2)不能传递大量参数

3)访问量高时会创建很多进程

3. curl 扩展

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

设置curl的超时时间 CURLOPT_TIMEOUT 为1 (最小为1),因此客户端需要等待1秒

代码如下:<?php

$ch = curl_init();

$curl_opt = array(

CURLOPT_URL, 'http://www.example.com/doRequest.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是最好的,缺点是需要自己拼接header部分。<?php

$url = 'http://www.example.com/doRequest.php';

$param = array(

'name'=>'fdipzone',

'gender'=>'male',

'age'=>30

);

doRequest($url, $param);

function doRequest($url, $param=array()){

$urlinfo = parse_url($url);

$host = $urlinfo['host'];

$path = $urlinfo['path'];

$query = isset($param)? http_build_query($param) : '';

$port = 80;

$errno = 0;

$errstr = '';

$timeout = 10;

$fp = fsockopen($host, $port, $errno, $errstr, $timeout);

$out = "POST ".$path." HTTP/1.1\r\n";

$out .= "host:".$host."\r\n";

$out .= "content-length:".strlen($query)."\r\n";

$out .= "content-type:application/x-www-form-urlencoded\r\n";

$out .= "connection:close\r\n\r\n";

$out .= $query;

fputs($fp, $out);

fclose($fp);

}

?>

注意:当执行过程中,客户端连接断开或连接超时,都会有可能造成执行不完整,因此需要加上ignore_user_abort(true); // 忽略客户端断开

set_time_limit(0); // 设置执行不超时

fsockopen支持socket编程,可以使用fsockopen实现邮件发送等socket程序等等,使用fcockopen需要自己手动拼接出header部分

可以参考: http://cn.php.net/fsockopen/

使用示例如下:$fp = fsockopen("www.34ways.com", 80, $errno, $errstr, 30);

if (!$fp) {

echo "$errstr ($errno)
\n";

} else {

$out = "GET /index.php / HTTP/1.1\r\n";

$out .= "Host: www.34ways.com\r\n";

$out .= "Connection: Close\r\n\r\n";

fwrite($fp, $out);

/*忽略执行结果

while (!feof($fp)) {

echo fgets($fp, 128);

}*/

fclose($fp);

}

所以总结来说,fscokopen()函数应该可以满足您的要求。可以尝试一下。

fscokopen的问题和popen 一样,并发非常多时会产生很多子进程,当达到apache的连接限制数时,就会挂掉,我问题已经说了这种情况。

PHP 本身没有多线程的东西,但可以曲线的办法来造就出同样的效果,比如多进程的方式来达到异步调用,只限于命令模式。还有一种更简单的方式,可用于 Web 程序中,那就是用fsockopen()、fputs() 来请求一个 URL 而无需等待返回,如果你在那个被请求的页面中做些事情就相当于异步了。

关键代码如下:$fp=fsockopen('localhost',80,&$errno,&$errstr,5);

if(!$fp){

echo "$errstr ($errno)
\n";

}

fputs($fp,"GET another_page.php?flag=1\r\n");

fclose($fp);

上面的代码向页面 another_page.php 发送完请求就不管了,用不着等待请求页面的响应数据,利用这一点就可以在被请求的页面 another_page.php 中异步的做些事情了。

比如,一个很切实的应用,某个 Blog 在每 Post 了一篇新日志后需要给所有它的订阅者发个邮件通知。如果按照通常的方式就是:日志写完 -> 点提交按钮 -> 日志插入到数据库 -> 发送邮件通知 ->

告知撰写者发布成功

那么作者在点提交按钮到看到成功提示之间可能会等待很常时间,基本是在等邮件发送的过程,比如连接邮件服务异常、或器缓慢或是订阅者太多。而实际上是不管邮件发送成功与否,保证日志保存成功基本可接受的,所以等待邮件发送的过程是很不经济的,这个过程可异步来执行,并且邮件发送的结果不太关心或以日志形式记录备查。

改进后的流程就是:日志写完 -> 点提交按钮 -> 日志插入到数据库 --->

告知撰写者发布成功

└ 发送邮件通知 -> [记下日志]

用个实际的程序来测试一下,有两个 php,分别是 write.php 和 sendmail.php,在 sendmail.php 用 sleep(seconds) 来模拟程序执行使用时间。

write.php,执行耗时 1 秒<?php

function asyn_sendmail() {

$fp=fsockopen('localhost',80,&$errno,&$errstr,5);

if(!$fp){

echo "$errstr ($errno)
\n";

}

sleep(1);

fputs($fp,"GET /sendmail.php?param=1\r\n"); #请求的资源 URL 一定要写对

fclose($fp);

}

echo time().'
';

echo 'call asyn_sendmail
';

asyn_sendmail();

echo time().'
';

?>

sendmail.php,执行耗时 10 秒

//sendmail();

//sleep 10 seconds

sleep(10);

fopen('C:\'.time(),'w');

?>

通过页面访问 write.php,页面输出:1272472697 call asyn_sendmail

1272472698

并且在 C:\ 生成文件:1272472708

从上面的结果可知 sendmail.php 花费至少 10 秒,但不会阻塞到 write.php 的继续往下执行,表明这一过程是异步的。

php后端异步处理数据,详解 PHP 异步后台处理相关推荐

  1. Android Loader 异步加载详解二:探寻Loader内部机制

    Android Loader 异步加载详解二:探寻Loader内部机制 转载请标明出处:http://blog.csdn.net/zhaoyanjun6/article/details/7025991 ...

  2. Android Loader 异步加载详解一:基础概念

    转载请标明出处:http://blog.csdn.net/zhaoyanjun6/article/details/70241844 本文出自[赵彦军的博客] Android Loader 异步加载详解 ...

  3. vue连接后端本地接口_详解vue配置后台接口方式

    详解vue配置后台接口方式 在vueAdmin项目中有两种方式配置后端接口的方式,在此做下记录 第一种(代理方式) 在工程目录下 > config > index.js - 修改为如下配置 ...

  4. dicom多帧转换_Python解析多帧dicom数据详解

    概述 pydicom是一个常用python DICOM parser.但是,没有提供解析多帧图的示例.本文结合相关函数和DICOM知识做一个简单说明. DICOM多帧数据存储 DICOM标准中关于多帧 ...

  5. android ble蓝牙接收不到数据_Android蓝牙4.0 Ble读写数据详解 -2

    Android蓝牙4.0 Ble读写数据详解 -2 上一篇说了如何扫描与链接蓝牙 这篇文章讲讲与蓝牙的数据传输,与一些踩到的坑. 先介绍一款调试工具,专门调试Ble蓝牙的app.名字叫:nRF-Con ...

  6. InheritableThreadLocal类原理简介使用 父子线程传递数据详解 多线程中篇(十八)...

    上一篇文章中对ThreadLocal进行了详尽的介绍,另外还有一个类: InheritableThreadLocal 他是ThreadLocal的子类,那么这个类又有什么作用呢? 测试代码 publi ...

  7. IoT:大端与小端字节数据详解

    大端与小端字节数据详解 转自:https://blog.csdn.net/dosthing/article/details/80641173 前言 计算机的数据以01构成的字节存储,这就涉及数据大小端 ...

  8. python爬取app中的音频_Python爬取喜马拉雅音频数据详解

    码农公社  210.net.cn  210是何含义?10月24日是程序员节,1024 =210.210既 210 之意. Python爬取喜马拉雅音频数据详解 一.项目目标 爬取喜马拉雅音频数据 受害 ...

  9. python能处理nc文件吗_利用python如何处理nc数据详解

    前言 这两天帮一个朋友处理了些 nc 数据,本以为很简单的事情,没想到里面涉及到了很多的细节和坑,无论是"知难行易"还是"知易行难"都不能充分的说明问题,还是& ...

最新文章

  1. Go 初体验 - 令人惊叹的语法 - defer.3 - defer 函数参数计算时机
  2. CenterWindow()的用法
  3. Linux搭建lamp(Apache+PHP+Mysql环境)centos7.2版详细教程
  4. 附加数据库时出错。有关详细信息,请单击“消息”列中的超链接。
  5. 剖析——移动构造函数
  6. C++设计模式之策略模式(Strategy)
  7. Oracle基本常用命令
  8. QT_TableWidget插入checkbox
  9. Gaze Estimation学习笔记(1)-Appearance-Based Gaze Estimation in the Wild
  10. vmware player 坑
  11. BOOST库介绍(五)——智能指针shared_ptr
  12. 水稻生物育种突破 国稻种芯-何登骥:功能性农业外源植物导入
  13. 世界顶级音响品牌排名
  14. 二叉树的中序遍历-python
  15. redis查看集合中元素的数量,scard
  16. Codeforces Round #322 B Luxurious Houses
  17. python简笔画程序_用python设计程序输生日判断星座,及星座简笔画。
  18. vscode中!无法生成模板解决方案
  19. 使用Scrapy爬取斗鱼图片
  20. 把项目从meeclipces转移到idead中遇见的问题

热门文章

  1. 雷军:创业起飞要找肥的行业等待台风口
  2. CAutoupdater通用自动升级组件
  3. my.资料收集_20170912
  4. 4.解决Hash冲突的几种方法
  5. 项目时间管理的四个步骤
  6. SQL去掉字段中的特殊字符
  7. 基于MapboxGL的样式文件自动生成图例
  8. js 监听手机输入法弹出
  9. python 生成html文档_Python使用pyh生成HTML文档的方法示例
  10. 孩子给离世父亲发短信被回复:“我的孩子是最棒的,爸爸也想你”