PHP非阻塞实现方法
为让PHP在后端处理长时间任务时不阻塞,快速响应页面请求,可以有如下措施:
1使用fastcgi_finish_request()
如果PHP与Web服务器使用了PHP-FPM(FastCGI进程管理器),那通过 fastcgi_finish_request() 函数能马上结束会话,而PHP线程可以继续在后台运行。
echo"programstart...";
file_put_contents('log.txt','start-time:'.date('Y-m-dH:i:s'),FILE_APPEND);
fastcgi_finish_request();
sleep(1);
echo'debug...';
file_put_contents('log.txt','start-proceed:'.date('Y-m-dH:i:s'),FILE_APPEND);
sleep(10);
file_put_contents('log.txt','end-time:'.date('Y-m-dH:i:s'),FILE_APPEND);
从输出结果可看到,页面打印完programstart...,输出第一行到log.txt后会话就返回了,所以后面的debug... 不会在浏览器上显示,而log.txt文件能完整地接收到三个完成时间。
2使用fsockopen()
使用 fsockopen()打开一个网络连接或者一个Unix套接字连接,再用 stream_set_blocking()非阻塞模式请求:
$fp=fsockopen("www.example.com",80,$errno,$errstr,30);
if(!$fp){
die('errorfsockopen');
}
//转换到非阻塞模式
stream_set_blocking($fp,0);
$http="GET/save.php /HTTP/1.1\r\n";
$http.="Host:www.example.com\r\n";
$http.="Connection:Close\r\n\r\n";
fwrite($fp,$http);
fclose($fp);
3使用cURL
利用cURL中的curl_multi_*函数发送异步请求
$mh=curl_multi_init(); $ch=curl_init(); curl_setopt($ch,CURLOPT_URL,"http://localhost/"); curl_multi_add_handle($mh,$ch); curl_multi_exec($mh,$active); curl_close($ch); curl_multi_remove_handle($mh,$ch); curl_multi_close($mh); echo"End\n";
4使用Gearman/Swoole扩展
Gearman是一个具有php扩展的分布式异步处理框架,能处理大批量异步任务。
Swoole最近很火,有很多异步方法,使用简单。
Swoole最近很火,有很多异步方法,使用简单。
5使用缓存和队列
使用redis等缓存、队列,将数据写入缓存,使用后台计划任务实现数据异步处理。
这个方法在常见的大流量架构中应该很常见吧
6调用系统命令
极端的情况下,可以调用系统命令,可以将数据传给后台任务执行,个人感觉不是很高效。
$cmd='nohupphp./processd.php$someVar>/dev/null &'; `$cmd`
7使用pcntl_fork()
安装pcntl扩展,使用pcntl_fork()生成子进程异步执行任务,个人觉得是最方便的,但也容易出现僵尸进程。
$pid=pcntl_fork()
if($pid==0){
child_func(); //子进程函数,主进程运行
}else{
father_func(); //主进程函数
}
echo"Process".getmypid()."gettotheend.\n";
functionfather_func(){
echo"Fatherpidis".getmypid()."\n";
}
functionchild_func(){
sleep(6);
echo"Childprocessexitpidis".getmypid()."\n";
exit(0);
}
8PHP原生支持
外国佬的大招,没看懂
http://nikic.github.io/2012/12/22/Cooperative-multitasking-using-coroutines-in-PHP.html
参考资料:
- PHP非阻塞模式