php 不等待返回的實現方法(異步調用)

PHP異步執行的常用方式常見的有以下幾種,可以根據各自優缺點進行選擇:

1.客戶端頁面採用AJAX技術請求服務器
優點
:最簡單,也最快,就是在返回給客戶端的HTML代碼中,嵌入AJAX調用,或者,嵌入一個img標籤,src指向要執行的耗時腳本。
缺點:一般來說Ajax都應該在onLoad以後觸發,也就是說,用戶點開頁面後,就關閉,那就不會觸發我們的後臺腳本了。
而使用img標籤的話,這種方式不能稱爲嚴格意義上的異步執行。用戶瀏覽器會長時間等待php腳本的執行完成,也就是用戶瀏覽器的狀態欄一直顯示還在load。
當然,還可以使用其他的類似原理的方法,比如script標籤等等。

2.popen()函數
該函數打開一個指向進程的管道,該進程由派生給定的 command 命令執行而產生。打開一個指向進程的管道,該進程由派生給定的 command 命令執行而產生。
所以可以通過調用它,但忽略它的輸出。使用代碼如下:

  1. pclose(popen("/home/xinchen/backend.php &", 'r'));

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

3.CURL擴展
CURL是一個強大的HTTP命令行工具,可以模擬POST/GET等HTTP請求,然後得到和提取數據,顯示在"標準輸出"(stdout)上面。代碼如下:

  1. $ch = curl_init();
  2.  
  3. $curl_opt = array(CURLOPT_URL, 'http://www.example.com/backend.php',
  4.                             CURLOPT_RETURNTRANSFER, 1,
  5.                             CURLOPT_TIMEOUT, 1,);
  6.  
  7. curl_setopt_array($ch, $curl_opt);
  8.  
  9. curl_exec($ch);
  10.  
  11. curl_close($ch);

缺點如你問題中描述的一樣,由於使用CURL需要設置CUROPT_TIMEOUT爲1(最小爲1,鬱悶)。也就是說,客戶端至少必須等待1秒鐘。

4.fscokopen()函數
fsockopen支持socket編程,可以使用fsockopen實現郵件發送等socket程序等等,使用fcockopen需要自己手動拼接出header部分
可以參考: http://cn.php.net/fsockopen/
使用示例如下:

  1. $fp = fsockopen("www.34ways.com", 80, $errno, $errstr, 30);
  2. if (!$fp) {
  3.     echo "$errstr ($errno)<br />\n";
  4. } else {
  5.     $out = "GET /index.php  / HTTP/1.1\r\n";
  6.     $out .= "Host: www.34ways.com\r\n";
  7.     $out .= "Connection: Close\r\n\r\n";
  8.  
  9.     fwrite($fp, $out);
  10.     /*忽略執行結果
  11.     while (!feof($fp)) {
  12.         echo fgets($fp, 128);
  13.     }*/
  14.     fclose($fp);
  15. }

所以總結來說,fscokopen()函數應該可以滿足您的要求。可以嘗試一下。

  • fscokopen的問題和popen 一樣,併發非常多時會產生很多子進程,當達到apache的連接限制數時,就會掛掉,我問題已經說了這種情況。


PHP 本身沒有多線程的東西,但可以曲線的辦法來造就出同樣的效果,比如多進程的方式來達到異步調用,只限於命令模式。還有一種更簡單的方式,可用於 Web 程序中,那就是用fsockopen()、fputs() 來請求一個 URL 而無需等待返回,如果你在那個被請求的頁面中做些事情就相當於異步了。

關鍵代碼如下:

  1. $fp=fsockopen('localhost',80,&$errno,&$errstr,5);
  2. if(!$fp){
  3.     echo "$errstr ($errno)<br />\n";
  4. }
  5. fputs($fp,"GET another_page.php?flag=1\r\n");
  6. fclose($fp);

上面的代碼向頁面 another_page.php 發送完請求就不管了,用不着等待請求頁面的響應數據,利用這一點就可以在被請求的頁面 another_page.php 中異步的做些事情了。

比如,一個很切實的應用,某個 Blog 在每 Post 了一篇新日誌後需要給所有它的訂閱者發個郵件通知。如果按照通常的方式就是:

日誌寫完 -> 點提交按鈕 -> 日誌插入到數據庫 -> 發送郵件通知 ->
告知撰寫者發佈成功

那麼作者在點提交按鈕到看到成功提示之間可能會等待很常時間,基本是在等郵件發送的過程,比如連接郵件服務異常、或器緩慢或是訂閱者太多。而實際上是不管郵件發送成功與否,保證日誌保存成功基本可接受的,所以等待郵件發送的過程是很不經濟的,這個過程可異步來執行,並且郵件發送的結果不太關心或以日誌形式記錄備查。

改進後的流程就是:

日誌寫完 -> 點提交按鈕 -> 日誌插入到數據庫 --->
告知撰寫者發佈成功
└ 發送郵件通知 -> [記下日誌]

用個實際的程序來測試一下,有兩個 php,分別是 write.php 和 sendmail.php,在 sendmail.php 用 sleep(seconds) 來模擬程序執行使用時間。

write.php,執行耗時 1 秒

  1. <?php
  2.  
  3. function asyn_sendmail() {
  4.     $fp=fsockopen('localhost',80,&$errno,&$errstr,5);
  5.     if(!$fp){
  6.         echo "$errstr ($errno)<br />\n";
  7.     }
  8.     sleep(1);
  9.     fputs($fp,"GET /sendmail.php?param=1\r\n"); #請求的資源 URL 一定要寫對
  10.     fclose($fp);
  11. }
  12.  
  13. echo time().'<br>';
  14. echo 'call asyn_sendmail<br>';
  15. asyn_sendmail();
  16. echo time().'<br>';
  17. ?>

sendmail.php,執行耗時 10 秒

  1. <?php
  2. //sendmail();
  3. //sleep 10 seconds
  4. sleep(10);
  5. fopen('C:\'.time(),'w');
  6. ?>

通過頁面訪問 write.php,頁面輸出:

1272472697 call asyn_sendmail
1272472698

並且在 C:\ 生成文件:

1272472708

從上面的結果可知 sendmail.php 花費至少 10 秒,但不會阻塞到 write.php 的繼續往下執行,表明這一過程是異步的。


轉自:http://www.dewen.io/q/3970/

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章