為讓 PHP 在後端處理長時間任務時不阻塞,快速響應頁面請求,可以有如下措施:
1 使用 fastcgi_finish_request()
如果 PHP 與 Web 服務器使用了 PHP-FPM(FastCGI 進程管理器),那通過 fastcgi_finish_request() 函數能馬上結束會話,而 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-d H:i:s'), FILE_APPEND);
從輸出結果可看到,頁面打印完program start…,輸出第一行到 log.txt 後會話就返回了,所以後面的 debug… 不會在瀏覽器上顯示,而 log.txt 文件能完整地接收到三個完成時間。
2 使用 fsockopen()
使用 fsockopen() 打開一個網絡連接或者一個Unix套接字連接,再用 stream_set_blocking() 非阻塞模式請求:
$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);
3 使用 cURL
利用cURL中的 curl_multi_* 函數發送異步請求
$cmh = curl_multi_init();
$ch1 = curl_init();
curl_setopt($ch1, CURLOPT_URL, "http://localhost/");
curl_multi_add_handle($cmh, $ch1);
curl_multi_exec($cmh, $active);
echo "End\n";
4 使用 Gearman/Swoole 擴展
Gearman 是一個具有 php 擴展的分佈式異步處理框架,能處理大批量異步任務。
Swoole 最近很火,有很多異步方法,使用簡單。
5 使用緩存和隊列
使用redis等緩存、隊列,將數據寫入緩存,使用後臺計劃任務實現數據異步處理。
這個方法在常見的大流量架構中應該很常見吧
6 調用系統命令
極端的情況下,可以調用系統命令,可以將數據傳給後臺任務執行,個人感覺不是很高效。
$cmd = 'nohup php ./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() . " 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);
}
8 PHP 原生支持
外國佬的大招,沒看懂
http://nikic.github.io/2012/12/22/Cooperative-multitasking-using-coroutines-in-PHP.html
歪麥博客
閱讀更多 生活社會大雜燴 的文章