by 尘缘 on 七月 31st, 2014 // Filed Under → php php
让PHP再也不阻塞当PHP做为后端处理须要完成一些长时间处理,为了快速响应页面请求,不做结果返回判断的状况下,能够有以下措施: html
1、若你使用的是FastCGI模式,使用fastcgi_finish_request()能立刻结束会话,但PHP线程继续在跑。 git
01
02
03
04
05
06
07
08
09
10
|
echo"program start.";
file_put_contents('log.txt','start-time:'.date('Y-m-d H:i:s'), FILE_APPEND);
fastcgi_finish_request();
sleep(1);
echo'debug...';
file_put_contents('log.txt','start-proceed:'.date('Y-m-d H:i:s'), FILE_APPEND);
sleep(10);
file_put_contents('log.txt','end-time:'.date('Y-m-d H:i:s'), FILE_APPEND);
|
这个例子输出结果可看到输出program start.后会话就返回了,因此debug那个输出浏览器是接收不到的,而log.txt文件能完整接收到三个完成时间。 github
2、使用fsockopen、cUrl的非阻塞模式请求另外的网址 redis
1
2
3
4
5
6
7
8
|
$fp=fsockopen("www.example.com", 80,$errno,$errstr, 30);
if(!$fp)die('error fsockopen');
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);
|
利用cURL中的curl_multi_*函数发送异步请求 后端
1
2
3
4
5
6
|
$cmh= curl_multi_init();
$ch1= curl_init();
curl_setopt($ch1, CURLOPT_URL,"http://localhost:6666/child.php");
curl_multi_add_handle($cmh,$ch1);
curl_multi_exec($cmh,$active);
echo"End\n";
|
3、使用Gearman、Swoole扩展
Gearman是一个具备php扩展的分布式异步处理框架,能处理大批量异步任务;
Swoole最近很火,有不少异步方法,使用简单。(尘缘注:号称从新定义PHP,把NodeJS喷得体无完肤。Swoole工具虽好,却感受是扩展自己跟NodeJS没可比性) 浏览器
4、使用redis等缓存、队列,将数据写入缓存,使用后台计划任务实现数据异步处理。
这个方法在常见的大流量架构中应该很常见吧 缓存
5、极端的状况下,能够调用系统命令,能够将数据传给后台任务执行,我的感受不是很高效。 架构
1
2
|
$cmd='nohup php ./processd.php $someVar >/dev/null &';
`$cmd`
|
6、外国佬的大招,没看懂,php原生支持 框架
http://nikic.github.io/2012/12/22/Cooperative-multitasking-using-coroutines-in-PHP.html
7、安装pcntl扩展,使用pcntl_fork生成子进程异步执行任务,我的以为是最方便的,但也容易出现zombie process。
01
02
03
04
05
06
07
08
09
10
11
12
13
14
15
16
17
|
if(($pid= pcntl_fork()) == 0) {
child_func(); //子进程函数,主进程运行
}else{
father_func(); //主进程函数
}
echo"Process ".getmypid() ." get to the end.\n";
functionfather_func() {
echo"Father pid is ".getmypid() ."\n";
}
functionchild_func() {
sleep(6);
echo"Child process exit pid is ".getmypid() ."\n";
exit(0);
}
|