讓PHP不再阻塞當PHP作為後端處理需要完成一些長時間處理,為了快速響應頁面請求,不作結果返回判斷的情況下,可以有如下措施:
一、若你使用的是FastCGI模式,使用fastcgi_finish_request()能馬上結束會話,但PHP線程繼續在跑。
幫助 01 02 03 04 05 06 07 08 09 10echo
"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文件能完整接收到三個完成時間。
二、使用fsockopen、cUrl的非阻塞模式請求另外的網址
幫助 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"
;
三、使用Gearman、Swoole擴展
Gearman是一個具有php擴展的分布式異步處理框架,能處理大批量異步任務;
Swoole最近很火,有很多異步方法,使用簡單。(塵緣注:號稱重新定義PHP,把NodeJS噴得體無完膚。Swoole工具雖好,卻感覺是擴展本身跟NodeJS沒可比性)
四、使用redis等緩存、隊列,將數據寫入緩存,使用後台計劃任務實現數據異步處理。
這個方法在常見的大流量架構中應該很常見吧
五、極端的情況下,可以調用系統命令,可以將數據傳給後台任務執行,個人感覺不是很高效。
幫助 1 2$cmd
=
'nohup php ./processd.php $someVar >/dev/null &'
;
`
$cmd
`
六、外國佬的大招,沒看懂,php原生支持
http://nikic.github.io/2012/12/22/Cooperative-multitasking-using-coroutines-in-PHP.html
七、安裝pcntl擴展,使用pcntl_fork生成子進程異步執行任務,個人覺得是最方便的,但也容易出現zombie process。
幫助 01 02 03 04 05 06 07 08 09 10 11 12 13 14 15 16 17if
((
$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);
}