PHP非阻塞模式

讓PHP再也不阻塞當PHP做爲後端處理須要完成一些長時間處理,爲了快速響應頁面請求,不做結果返回判斷的狀況下,能夠有以下措施:php

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

這個例子輸出結果可看到輸出program start.後會話就返回了,因此debug那個輸出瀏覽器是接收不到的,而log.txt文件能完整接收到三個完成時間。html

2、使用fsockopen、cUrl的非阻塞模式請求另外的網址git

$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_*函數發送異步請求github

$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沒可比性)redis

4、使用redis等緩存、隊列,將數據寫入緩存,使用後臺計劃任務實現數據異步處理。
這個方法在常見的大流量架構中應該很常見吧後端

5、極端的狀況下,能夠調用系統命令,能夠將數據傳給後臺任務執行,我的感受不是很高效。瀏覽器

$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。架構

if (( $pid = pcntl_fork()) == 0) {
     child_func();    //子進程函數,主進程運行
} else {
     father_func();   //主進程函數
}
 
echo "Process " . getmypid () . " get to the end.\n" ;
 
function father_func() {
     echo "Father pid is " . getmypid () . "\n" ;
}
 
function child_func() {
     sleep(6);
     echo "Child process exit pid is " . getmypid () . "\n" ;
     exit (0);
}
相關文章
相關標籤/搜索