公司**一直採用著非同步呼叫,好處挺多。
瀏覽器和伺服器之間是通過 http 協議進行連線通訊的。這是一種基於請求和響應模型的協議。瀏覽器通過 url 向伺服器發起請求,web 伺服器接收到請求,執行一段程式,然後做出響應,傳送相應的html**給客戶端。
這就有了乙個問題,web 伺服器執行一段程式,可能幾毫秒就完成,也可能幾分鐘都完不成。如果程式執行緩慢,使用者可能沒有耐心等下去,就關閉瀏覽器了。
而有的時候,我們更本不關心這些耗時的指令碼的返回結果,但卻還要等他執行完返回,才能繼續下一步。
那麼有沒有什麼辦法,只是簡單的觸發呼叫這些耗時的指令碼然後就繼續下一步,讓這些耗時的指令碼在服務端慢慢執行?
經過試驗,總結出來幾種方法,和大家share:
1. 最簡單的辦法,就是在返回給客戶端的html**中,嵌入ajax呼叫,或者,嵌入乙個img標籤,src指向要執行的耗時指令碼。
這種方法最簡單,也最快。伺服器端不用做任何的呼叫。
但是缺點是,一般來說ajax都應該在onload以後觸發,也就是說,使用者點開頁面後,就關閉,那就不會觸發我們的後台指令碼了。
而使用img標籤的話,這種方式不能稱為嚴格意義上的非同步執行。使用者瀏覽器會長時間等待php指令碼的執行完成,也就是使用者瀏覽器的狀態列一直顯示還在load。
當然,還可以使用其他的類似原理的方法,比如script標籤等等。
2. popen()
resource popen ( string command, string mode );
//開啟乙個指向程序的管道,該程序由派生給定的 command 命令執行而產生。開啟乙個指向程序的管道,該程序由派生給定的 command 命令執行而產生。
所以可以通過呼叫它,但忽略它的輸出。
pclose(popen("/home/xinchen/backend.php &
", '
r'));
這個方法避免了第乙個方法的缺點,並且也很快。但是問題是,這種方法不能通過http協議請求另外的乙個webservice,只能執行本地的指令碼檔案。並且只能單向開啟,無法穿大量引數給被呼叫指令碼。
並且如果,訪問量很高的時候,會產生大量的程序。如果使用到了外部資源,還要自己考慮競爭。
3. 使用curl
這個方法,設定curopt_timeout為1(最小為1,鬱悶)。也就是說,客戶端至少必須等待1秒鐘。
function curl_get($url,$second=30)
$url = '
';for($i=0;$i<4;$i++)
exit(
'資料採集開始
');
另外個test1.php檔案中**如下:
sleep(10); file_put_contents(
'd:/test.txt
','wolaile
' echo
"string
";
執行起來,可以看到很快就列印出了資料採集開始。
10秒之後,才能在d盤看到檔案的生成
4. 使用fsockopen
這個方法應該是最完美的,但是缺點是,你需要自己拼出http的header部分。
$fp = fsockopen("www.example.com
", 80, $errno, $errstr, 30
);
if (!$fp)
else
*/fclose($fp);
}
所以,總體來看,最好用,最簡單的還是第一種方法。
最完美的應該是最後一種,但是比較複雜
PHP實現非同步呼叫方法研究
瀏覽器和伺服器之間是通過 http 協議進行連線通訊的。這是一種基於請求和響應模型的協議。瀏覽器通過 url 向伺服器發起請求,web 伺服器接收到請求,執行一段程式,然後做出響應,傳送相應的html 給客戶端。這就有了乙個問題,web 伺服器執行一段程式,可能幾毫秒就完成,也可能幾分鐘都完不成。如...
PHP實現非同步呼叫方法研究
瀏覽器和伺服器之間是通過 http 協議進行連線通訊的。這是一種基於請求和響應模型的協議。瀏覽器通過 url 向伺服器發起請求,web 伺服器接收到請求,執行一段程式,然後做出響應,傳送相應的html 給客戶端。這就有了乙個問題,web 伺服器執行一段程式,可能幾毫秒就完成,也可能幾分鐘都完不成。如...
PHP實現非同步呼叫方法研究與分享
這樣就出現了乙個問題,乙個客戶端的相應服務端可能執行1秒也有可能執行1分鐘,這樣瀏覽器就會一直處於等待狀態,如果程式執行緩慢,使用者可能就沒耐心關掉了瀏覽器。而有的時候我們不需要關心程式執行的結果,沒有必要這樣浪費時間和耐心等待,那我們就要想出辦法讓程式不收等待在後台靜默執行。比如現在有乙個場景,給...