非阻塞模式是指利用socket事件的訊息機制,Server端與Client端之間的通訊處於非同步狀態。
讓PHP不再阻塞當PHP作為後端處理需要完成一些長時間處理,為了快速響應頁面請求,不作結果返回判斷的情況下,可以有如下措施:
一、若你使用的是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.後會話就返回了,所以debug那個輸出瀏覽器是接收不到的,而log.txt檔案能完整接收到三個完成時間。
二、使用fsockopen、cUrl的非阻塞模式請求另外的網址
$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_*函數發送非同步請求
$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等緩衝、隊列,將資料寫入緩衝,使用後台計劃任務實現資料非同步處理。
這個方法在常見的大流量架構中應該很常見吧
五、極端的情況下,可以調用系統命令,可以將資料傳給背景工作執行,個人感覺不是很高效。
$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。
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);}
以上內容是小編給大家介紹的PHP非阻塞模式,希望對大家有所協助!