PHP 實現異步數據調用
瀏覽器和服務器之間只有一種面向無連接的HTTP協議進行通訊的,面向無連接的程序的特點是客戶端請求服務端,服務端根據請求返回相應的程序,不能保持持久連接。這樣就出現了一個問題,一個客戶端的相應服務端可能執行1秒也有可能執行1分鐘,這樣瀏覽器就會一直處于等待狀態,如果程序執行緩慢,用戶可能就沒耐心關掉了瀏覽器。
而有的時候我們不需要關心程序執行的結果,沒有必要這樣浪費時間和耐心等待,那我們就要想出辦法讓程序不收等待在后臺靜默執行。比如現在有一個場景,給1000個用戶發送一封推薦郵件,用戶輸入或者導入郵件賬號了提交服務器執行發送。
<?php $count = count($emailarr);for($i =0; $i < $count; $i++) { sendmail();//發送郵件 } ?>
這段代碼用戶體驗極差,也無法實際運用,首先發送這么多郵件會產生服務器運行超時,其實漫長的用戶等待時間會讓用戶對系統產品懷疑和失去信心。但是用戶不需要等待到1000封郵件都發送完畢了才提交發送成功,我們完全可以提交后臺后直接給用戶提示發送成功,然后讓后臺程序靜默依次發送。
這個時候我們就需要“異步執行”技術來執行代碼,異步執行的特點是后臺靜默執行,用戶無需等待代碼的執行結果,使用異步執行的好處:
擺脫了應用程序對單個任務的依賴性提高了程序的執行效率提高了程序的擴展性在一定場景提高了用戶體驗因為PHP不支持多線程,使用異步調用的請求多個HTTP的方式達到了程序并行執行效果,但是注意的是請求的HTTP過多的話,會大大加大了系統的開銷。PHP異步執行的常用方式1. 客戶端頁面采用AJAX技術請求服務器
最簡單的辦法就是在返回給客戶端的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擴展
CURL是一個強大的HTTP命令行工具,可以模擬POST/GET等HTTP請求,然后得到和提取數據,顯示在'標準輸出'(stdout)上面。
$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);
使用CURL需要設置CUROPT_TIMEOUT為1(最小為1,郁悶)。也就是說,客戶端至少必須等待1秒鐘。
4. fscokopen()函數
fsockopen是一個非常強大的函數,支持socket編程,可以使用fsockopen實現郵件發送等socket程序等等,使用fcockopen需要自己手動拼接出header部分。官方文檔: http://cn.php.net/fsockopen/。
$fp = fsockopen('www.example.com', 80, $errno, $errstr, 30);if (!$fp) { echo '$errstr ($errno)<br />n';} else { $out = 'GET /backend.php / HTTP/1.1rn'; $out .= 'Host: www.example.comrn'; $out .= 'Connection: Closernrn'; fwrite($fp, $out); /*忽略執行結果 while (!feof($fp)) {echo fgets($fp, 128); }*/ fclose($fp);}
相關文章:
