為讓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_* 函數發送非同步請求
$mh = curl_multi_init(); $ch = curl_init(); curl_setopt($ch, CURLOPT_URL, "http://localhost/"); curl_multi_add_handle($mh, $ch); curl_multi_exec($mh, $active); curl_close($ch); curl_multi_remove_handle($mh, $ch); curl_multi_close($mh); 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); }
以上是PHP非同步非阻塞實作方法的詳細內容。更多資訊請關注PHP中文網其他相關文章!