PHP非阻塞模式

PHP非阻塞模式

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);
}
相關文章
相關標籤/搜索