PHP異步調(diào)用實(shí)現(xiàn)方式
瀏覽器和服務(wù)器之間只一種面向無(wú)連接的HTTP協(xié)議進(jìn)行通訊的,面向無(wú)連接的程序的特點(diǎn)是客戶端請(qǐng)求服務(wù)端,服務(wù)端根據(jù)請(qǐng)求輸出相應(yīng)的程序,不能保持持久連接。
這樣就出現(xiàn)了一個(gè)問(wèn)題,一個(gè)客戶端的相應(yīng)服務(wù)端可能執(zhí)行1秒也有可能執(zhí)行1分鐘,這樣瀏覽器就會(huì)一直處于等待狀態(tài),如果程序執(zhí)行緩慢,用戶可能就沒(méi)耐心關(guān)掉了瀏覽器。
而有的時(shí)候我們不需要關(guān)心程序執(zhí)行的結(jié)果,沒(méi)有必要這樣浪費(fèi)時(shí)間和耐心等待,那我們就要想出辦法讓程序不收等待在后臺(tái)靜默執(zhí)行。
比如現(xiàn)在有一個(gè)場(chǎng)景,給1000個(gè)用戶發(fā)送一封推薦郵件,用戶輸入或者導(dǎo)入郵件賬號(hào)了提交服務(wù)器執(zhí)行發(fā)送。
復(fù)制代碼代碼如下:
<?php
$count=count($emailarr);
for($i=0;$i<$count;$i++)
{
sendmail(.....);//發(fā)送郵件
}
?>
這段代碼用戶體驗(yàn)極差,也無(wú)法實(shí)際運(yùn)用,首先發(fā)送這么多郵件會(huì)產(chǎn)生服務(wù)器運(yùn)行超時(shí),其實(shí)漫長(zhǎng)的用戶等待時(shí)間會(huì)讓用戶對(duì)系統(tǒng)產(chǎn)品懷疑和失去信心。但是用戶不需要等待到1000封郵件都發(fā)送完畢了才提交發(fā)送成功,我們完全可以提交后臺(tái)后直接給用戶提示發(fā)送成功,然后讓后臺(tái)程序靜默依次發(fā)送。
這個(gè)時(shí)候我們就需要“異步執(zhí)行”技術(shù)來(lái)執(zhí)行代碼,異步執(zhí)行的特點(diǎn)是后臺(tái)靜默執(zhí)行,用戶無(wú)需等待代碼的執(zhí)行結(jié)果,使用異步執(zhí)行的好處:
1.擺脫了應(yīng)用程序?qū)蝹€(gè)任務(wù)的依賴性
2.提高了程序的執(zhí)行效率
3.提高了程序的擴(kuò)展性
4.在一定場(chǎng)景提高了用戶體驗(yàn)
5.因?yàn)镻HP不支持多線程,使用異步調(diào)用的請(qǐng)求多個(gè)HTTP的方式達(dá)到了程序并行執(zhí)行效果,但是注意的是請(qǐng)求的HTTP過(guò)多的話,會(huì)大大加大了系統(tǒng)的開(kāi)銷
PHP異步執(zhí)行的常用方式:
1.客戶端頁(yè)面采用AJAX技術(shù)請(qǐng)求服務(wù)器
1. 最簡(jiǎn)單的辦法,就是在返回給客戶端的HTML代碼中,嵌入AJAX調(diào)用,或者,嵌入一個(gè)img標(biāo)簽,src指向要執(zhí)行的耗時(shí)腳本。
這種方法最簡(jiǎn)單,也最快。服務(wù)器端不用做任何的調(diào)用。
但是缺點(diǎn)是,一般來(lái)說(shuō)Ajax都應(yīng)該在onLoad以后觸發(fā),也就是說(shuō),用戶點(diǎn)開(kāi)頁(yè)面后,就關(guān)閉,那就不會(huì)觸發(fā)我們的后臺(tái)腳本了。
而使用img標(biāo)簽的話,這種方式不能稱為嚴(yán)格意義上的異步執(zhí)行。用戶瀏覽器會(huì)長(zhǎng)時(shí)間等待php腳本的執(zhí)行完成,也就是用戶瀏覽器的狀態(tài)欄一直顯示還在load。
當(dāng)然,還可以使用其他的類似原理的方法,比如script標(biāo)簽等等
2.popen()函數(shù)
resource popen ( string command, string mode );
//打開(kāi)一個(gè)指向進(jìn)程的管道,該進(jìn)程由派生給定的 command 命令執(zhí)行而產(chǎn)生。打開(kāi)一個(gè)指向進(jìn)程的管道,該進(jìn)程由派生給定的 command 命令執(zhí)行而產(chǎn)生。
所以可以通過(guò)調(diào)用它,但忽略它的輸出。
pclose(popen("/home/xinchen/backend.php &", 'r'));
這個(gè)方法避免了第一個(gè)方法的缺點(diǎn),并且也很快。但是問(wèn)題是,這種方法不能通過(guò)HTTP協(xié)議請(qǐng)求另外的一個(gè)WebService,只能執(zhí)行本地的腳本文件。并且只能單向打開(kāi),無(wú)法穿大量參數(shù)給被調(diào)用腳本。
并且如果,訪問(wèn)量很高的時(shí)候,會(huì)產(chǎn)生大量的進(jìn)程。如果使用到了外部資源,還要自己考慮競(jìng)爭(zhēng)。
3.CURL擴(kuò)展
CURL是一個(gè)強(qiáng)大的HTTP命令行工具,可以模擬POST/GET等HTTP請(qǐng)求,然后得到和提取數(shù)據(jù),顯示在"標(biāo)準(zhǔn)輸出"(stdout)上面
復(fù)制代碼代碼如下:
$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需要設(shè)置CUROPT_TIMEOUT為1(最小為1,郁悶)。也就是說(shuō),客戶端至少必須等待1秒鐘。
4.fscokopen()函數(shù)
fsockopen是一個(gè)非常強(qiáng)大的函數(shù),支持socket編程,可以使用fsockopen實(shí)現(xiàn)郵件發(fā)送等socket程序等等,使用fcockopen需要自己手動(dòng)拼接出header部分
官方文檔: http://cn.php.net/fsockopen/
復(fù)制代碼代碼如下:
$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);
}
PHP異步調(diào)用socket
復(fù)制代碼代碼如下:
<?
$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); //將請(qǐng)求寫(xiě)入socket
//也可以選擇獲取server端的響應(yīng)
/*while (!feof($fp)) {
echo fgets($fp, 128);
}*/
//如果不等待server端響應(yīng)直接關(guān)閉socket即可
fclose($fp);
?>