程式非阻塞模式,這裡也可以理解成並發,為了讓PHP不再阻塞當PHP作為後端處理需要完成一些長時間處理,為了快速響應頁面請求,不作結果返回判斷的情況下,可以有如下措施:
具體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