PHP中實現非同步呼叫多線程程式碼

來源:互聯網
上載者:User

比如現在有一個情境,給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);

}

聯繫我們

該頁面正文內容均來源於網絡整理,並不代表阿里雲官方的觀點,該頁面所提到的產品和服務也與阿里云無關,如果該頁面內容對您造成了困擾,歡迎寫郵件給我們,收到郵件我們將在5個工作日內處理。

如果您發現本社區中有涉嫌抄襲的內容,歡迎發送郵件至: info-contact@alibabacloud.com 進行舉報並提供相關證據,工作人員會在 5 個工作天內聯絡您,一經查實,本站將立刻刪除涉嫌侵權內容。

A Free Trial That Lets You Build Big!

Start building with 50+ products and up to 12 months usage for Elastic Compute Service

  • Sales Support

    1 on 1 presale consultation

  • After-Sales Support

    24/7 Technical Support 6 Free Tickets per Quarter Faster Response

  • Alibaba Cloud offers highly flexible support services tailored to meet your exact needs.