首頁 >後端開發 >php教程 >php實作異步呼叫多執行緒的方法

php實作異步呼叫多執行緒的方法

黄舟
黄舟原創
2017-11-11 13:49:075530瀏覽

在之前的文章中我們先跟大家介紹php多執行緒的實作方法、那麼今天我們先給大家詳細介紹一下php實作非同步呼叫多執行緒的方法,希望透過這篇文章讓你對php多執行緒的了解進一步進階!

例如現在有一個場景,給1000個使用者發送一封推薦郵件,使用者輸入或匯入郵件帳號了提交伺服器執行發送

第一種解決方法:

<?php
$count=count($emailarr);
for($i=0;$i<$count;$i )
{
sendmail(.....);//发送邮件
}
?>

 這段程式碼使用者體驗極差,也無法實際運用,先發送這麼多郵件會產生伺服器運作逾時,其實漫長的使用者等待時間會讓使用者對系統產品懷疑和失去信心。但是用戶不需要等待到1000封郵件都發送完畢了才提交發送成功,我們完全可以提交後台後直接給用戶提示發送成功,然後讓後台程序靜默依次發送。

第二種解決方法:


#  我們嘗試使用「非同步執行」技術來執行程式碼,非同步執行的特徵是後台靜默執行,使用者無需等待程式碼的執行結果,使用非同步執行的好處:

1.擺脫了應用程式對單一任務的依賴性

2.提高了程式的執行效率

3.提高了程式的擴展性

4.在一定場景提高了用戶體驗

5.因為PHP不支援多線程,使用非同步呼叫的請求多個HTTP的方式達到了程式並行執行效果,但是注意的是請求的HTTP過多的話,會大大加大了系統的開銷

用戶體驗:用戶等待->發送完畢 
#然後把寄信任務轉給了一個單獨處理髮信的php程式處理了,當用戶看見「發送完畢」的時候其實信還沒寄完,這個時候,發信程式正在後台努力的工作著,一封一封的向外發送

<?php 
$domain="www.***.com"; 
$url="/system_mail.php"; 
$par="email=".implode(&#39;,&#39;,$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 &#39;&#39;发送完毕&#39;; 
?> 
system_mail.php 
<?php 
ini_set("ignore_user_abort",true); 
ignore_user_abort(true);//此处的代码需要php.ini开启相关的选项,保证php执行不超时的,不明白,参考我的另一篇文章 “关闭浏览器后,php脚本会不会继续运行” 
//获取email地址,发信,此处为发信代码 
?>

第三種方法:

1.最簡單的辦法,就是在傳回給客戶端的HTML程式碼中,嵌入AJAX調用,或者,嵌入一個img標籤,src指向要執行的耗時腳本。 
這種方法最簡單,也最快。伺服器端不用做任何的呼叫。 
但是缺點是,一般來說Ajax都應該在onLoad以後觸發,也就是說,用戶點開頁面後,就關閉,那就不會觸發我們的後台腳本了。 
而使用img標籤的話,這種方式不能稱為嚴格意義上的非同步執行。使用者瀏覽器會長時間等待php腳本的執行完成,也就是使用者瀏覽器的狀態欄一直顯示還在load。 
當然,也可以使用其他的類似原理的方法,例如script標籤等等。

第四種方法:


#popen()

resource popen ( string command, string mode );

//開啟一個指向進程的管道,該進程由派生給定的command 命令執行而產生。開啟一個指向進程的管道,該進程由派生給定的 command 命令執行而產生。

所以可以透過呼叫它,但忽略它的輸出。

pclose(popen("/home/xinchen/backend.php &", &#39;r&#39;));

這個方法避免了第一個方法的缺點,而且也很快。但問題是,這種方法不能透過HTTP協定請求另外的一個WebService,只能執行本地的腳本檔案。並且只能單向打開,無法穿著大量參數給被呼叫腳本。 
而且如果,訪問量很高的時候,會產生大量的進程。如果使用到了外部資源,還要自己考慮競爭。

第5種方法:

使用CURL 
這個方法,設定CUROPT_TIMEOUT為1(最小為1,鬱悶)。也就是說,客戶端至少必須等待1秒鐘。

$ch = curl_init();
$curl_opt = array(CURLOPT_URL, &#39;http://www.example.com/backend.php&#39;,
                            CURLOPT_RETURNTRANSFER, 1,
                            CURLOPT_TIMEOUT, 1,);
curl_setopt_array($ch, $curl_opt);
curl_exec($ch);
curl_close($ch);

第六種方法:

使用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);
}

總結:

以上六種方法就是實作php中非同步呼叫多執行緒的實例,每個方法都有它的特色,可以依照自己的需求來選擇適合自己的方法!

相關推薦;

php多執行緒模擬實作的三種方法介紹


php多重執行緒的實作實例


#php多執行緒一種實作方法—shell


PHP多執行緒小案例

#

以上是php實作異步呼叫多執行緒的方法的詳細內容。更多資訊請關注PHP中文網其他相關文章!

陳述:
本文內容由網友自願投稿,版權歸原作者所有。本站不承擔相應的法律責任。如發現涉嫌抄襲或侵權的內容,請聯絡admin@php.cn