瀏覽器和服務(wù)器之間是通過 HTTP 協(xié)議進行連接通訊的。這是一種基于請求和響應(yīng)模型的協(xié)議。瀏覽器通過 URL 向服務(wù)器發(fā)起請求,Web 服務(wù)器接收到請求,執(zhí)行一段程序,然后做出響應(yīng),發(fā)送相應(yīng)的html代碼給客戶端。
這就有了一個問題,Web 服務(wù)器執(zhí)行一段程序,可能幾毫秒就完成,也可能幾分鐘都完不成。如果程序執(zhí)行緩慢,用戶可能沒有耐心等下去,就關(guān)閉瀏覽器了。
而有的時候,我們更本不關(guān)心這些耗時的腳本的返回結(jié)果,但卻還要等他執(zhí)行完返回,才能繼續(xù)下一步。
那么有沒有什么辦法,只是簡單的觸發(fā)調(diào)用這些耗時的腳本然后就繼續(xù)下一步,讓這些耗時的腳本在服務(wù)端慢慢執(zhí)行?
經(jīng)過試驗,總結(jié)出來幾種方法,和大家share:
1. 最簡單的辦法,就是在返回給客戶端的HTML代碼中,嵌入AJAX調(diào)用,或者,嵌入一個img標(biāo)簽,src指向要執(zhí)行的耗時腳本。
這種方法最簡單,也最快。服務(wù)器端不用做任何的調(diào)用。
但是缺點是,一般來說Ajax都應(yīng)該在onLoad以后觸發(fā),也就是說,用戶點開頁面后,就關(guān)閉,那就不會觸發(fā)我們的后臺腳本了。
而使用img標(biāo)簽的話,這種方式不能稱為嚴(yán)格意義上的異步執(zhí)行。用戶瀏覽器會長時間等待php腳本的執(zhí)行完成,也就是用戶瀏覽器的狀態(tài)欄一直顯示還在load。
當(dāng)然,還可以使用其他的類似原理的方法,比如script標(biāo)簽等等。
2. popen()
resource popen ( string command, string mode );//打開一個指向進程的管道,該進程由派生給定的 command 命令執(zhí)行而產(chǎn)生。打開一個指向進程的管道,該進程由派生給定的 command 命令執(zhí)行而產(chǎn)生。
所以可以通過調(diào)用它,但忽略它的輸出。
pclose(popen("/home/xinchen/backend.php &", 'r'));
這個方法避免了第一個方法的缺點,并且也很快。但是問題是,這種方法不能通過HTTP協(xié)議請求另外的一個WebService,只能執(zhí)行本地的腳本文件。并且只能單向打開,無法穿大量參數(shù)給被調(diào)用腳本。
并且如果,訪問量很高的時候,會產(chǎn)生大量的進程。如果使用到了外部資源,還要自己考慮競爭。
3. 使用CURL
這個方法,設(shè)置CUROPT_TIMEOUT為1(最小為1,郁悶)。也就是說,客戶端至少必須等待1秒鐘。
$ch = curl_init(); $curl_opt = array(CURLOPT_URL, 'http://www.example.com/backend.php', CURLOPT_RETURNTRANSFER, 1, CURLOPT_TIMEOUT, 1,); curl_setopt_array($ch, $curl_opt); curl_exec($ch); curl_close($ch);
4. 使用fsockopen
這個方法應(yīng)該是最完美的,但是缺點是,你需要自己拼出HTTP的header部分。
$fp = fsockopen("www.example.com", 80, $errno, $errstr, 30); if (!$fp) { echo "$errstr ($errno)<br />/n"; } else { $out = "GET /backend.php / HTTP/1.1/r/n"; $out .= "Host: www.example.com/r/n"; $out .= "Connection: Close/r/n/r/n"; fwrite($fp, $out); /*忽略執(zhí)行結(jié)果 while (!feof($fp)) { echo fgets($fp, 128); }*/ fclose($fp); }
所以,總體來看,最好用,最簡單的還是第一種方法。
最完美的應(yīng)該是最后一種,但是比較復(fù)雜。
以上就是4種php實現(xiàn)異步調(diào)用的方法,希望對大家的學(xué)習(xí)有所幫助。