比如現在有一個情境,給1000個使用者發送一封推薦郵件,使用者輸入或者匯入郵件帳號了提交伺服器執行發送。
| 代碼如下 |
複製代碼 |
<?php $sqlserver/42852.htm target=_blank >count=count($emailarr); for($i=0;$i<$count;$i ) { sendmail(.....);//發送郵件 } ?> |
這段代碼使用者體驗極差,也無法實際運用,首先發送這麼多郵件會產生伺服器運行逾時,其實漫長的使用者等待時間會讓使用者對系統產品懷疑和失去信心。但是使用者不需要等待到1000封郵件都發送完畢了才提交發送成功,我們完全可以提交後台後直接給使用者提示發送成功,然後讓背景程式靜默依次發送。
這個時候我們就需要“非同步執行”技術來執行代碼,非同步執行的特點是後台靜默執行,使用者無需等待代碼的執行結果,使用非同步執行的好處:
1.擺脫了應用程式對單個任務的依賴性
2.提高了程式的執行效率
3.提高了程式的擴充性
4.在一定情境提高了使用者體驗
5.因為PHP不支援多線程,使用非同步呼叫的請求多個HTTP的方式達到了程式並存執行效果,但是注意的是請求的HTTP過多的話,會大大加大了系統的開銷
使用者體驗:使用者等待->發送完畢
朋友們就會問,怎麼缺少發信環節?
OK,發信環節就在使用者提交請求的時候,把發信任務轉給了一個單獨處理髮信的php程式處理了,當使用者看見“發送完畢”的時候其實信還沒發送完,這個時候,發信程式正在後台努力的工作著,一封一封的向外發送
sendmail.php
| 代碼如下 |
複製代碼 |
<?php $domain="www.***.com"; $url="/system_mail.php"; $par="email=".implode(',',$emailarr)."&........"; $header = "POST $url HTTP/1.0rn"; $header .= "Content-Type: application/x-www-form-urlencodedrn"; $header .= "Content-Length: " . strlen($par) . "rnrn"; $fp = @fsockopen ($domain, 80, $errno, $errstr, 30); fputs ($fp, $header . $par); fclose($fp); echo ''發送完畢'; ?> system_mail.php <?php ini_set("ignore_user_abort",true); ignore_user_abort(true);//此處的代碼需要php.ini開啟相關的選項,保證php執行不逾時的,不明白,參考我的另一篇文章 “關閉瀏覽器後,php指令碼會不會繼續運行” //擷取email地址,發信,此處為發信代碼 ?> |
好了,改成非同步方式後,使用者提交資訊,可以立即得到結果“發送完畢”。信呢,會在後台一封一封的發送,直到發送完畢。
經過實驗,總結出來幾種方法,和大家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秒鐘。
| 代碼如下 |
複製代碼 |
$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
這個方法應該是最完美的,但是缺點是,你需要自己拼出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.1rn"; $out .= "Host: www.example.comrn"; $out .= "Connection: Closernrn"; fwrite($fp, $out); /*忽略執行結果 while (!feof($fp)) { echo fgets($fp, 128); }*/ fclose($fp); } |