浏覽器和伺服器之間是通過 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'));</SPAN&lt; li>
這個方法避免了第一個方法的缺點,并且也很快。但是問題是,這種方法不能通過HTTP協定請求另外的一個WebService,隻能執行本地的腳本檔案。并且隻能單向打開,無法穿大量參數給被調用腳本。
并且如果,通路量很高的時候,會産生大量的程序。如果使用到了外部資源,還要自己考慮競争。
3. 使用CURL
這個方法,設定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);</SPAN&lt; li>
4. 使用fsockopen
這個方法應該是最完美的,但是缺點是,你需要自己拼出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);
/*忽略執行結果
while (!feof($fp)) {
echo fgets($fp, 128);
}*/
fclose($fp);
}</SPAN&lt; li>
是以,總體來看,最好用,最簡單的還是第一種方法。
最完美的應該是最後一種,但是比較複雜
如果有更好的辦法,歡迎交流。
POST:http://www.laruence.com/2008/04/14/318.html