PHP非同步呼叫實現方式
瀏覽器和伺服器之間只一種面向不需連線的HTTP協議進行通訊的,面向不需連線的程式的特點是用戶端請求服務端,服務端根據請求輸出相應的程式,不能保持持久串連。
這樣就出現了一個問題,一個用戶端的相應服務端可能執行1秒也有可能執行1分鐘,這樣瀏覽器就會一直處於等待狀態,如果程式執行緩慢,使用者可能就沒耐心關掉了瀏覽器。
而有的時候我們不需要關心程式執行的結果,沒有必要這樣浪費時間和耐心等待,那我們就要想出辦法讓程式不收等待在後台靜默執行。
比如現在有一個情境,給1000個使用者發送一封推薦郵件,使用者輸入或者匯入郵件帳號了提交伺服器執行發送。
複製代碼 代碼如下:
<?php
$count=count($emailarr);
for($i=0;$i<$count;$i++)
{
sendmail(.....);//發送郵件
}
?>
這段代碼使用者體驗極差,也無法實際運用,首先發送這麼多郵件會產生伺服器運行逾時,其實漫長的使用者等待時間會讓使用者對系統產品懷疑和失去信心。但是使用者不需要等待到1000封郵件都發送完畢了才提交發送成功,我們完全可以提交後台後直接給使用者提示發送成功,然後讓背景程式靜默依次發送。
這個時候我們就需要“非同步執行”技術來執行代碼,非同步執行的特點是後台靜默執行,使用者無需等待代碼的執行結果,使用非同步執行的好處:
1.擺脫了應用程式對單個任務的依賴性
2.提高了程式的執行效率
3.提高了程式的擴充性
4.在一定情境提高了使用者體驗
5.因為PHP不支援多線程,使用非同步呼叫的請求多個HTTP的方式達到了程式並存執行效果,但是注意的是請求的HTTP過多的話,會大大加大了系統的開銷
PHP非同步執行的常用方式:
1.用戶端頁面採用AJAX技術請求伺服器
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擴充
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.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);
}
PHP非同步呼叫socket
複製代碼 代碼如下:
<?
$host = "www.aaa.com";
$path = "/Report.php?ReportID=1";
$cookie = Session_id();
$fp = fsockopen($host, 80, $errno, $errstr, 30);
if (!$fp) {
print "$errstr ($errno)<br />\n";
exit;
}
$out = "GET ".$path." HTTP/1.1\r\n";
$out .= "Host: ".$host."\r\n";
$out .= "Connection: Close\r\n";
$out .= "Cookie: ".$cookie."\r\n\r\n";
fwrite($fp, $out); //將請求寫入socket
//也可以選擇擷取server端的響應
/*while (!feof($fp)) {
echo fgets($fp, 128);
}*/
//如果不等待server端響應直接關閉socket即可
fclose($fp);
?>