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


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

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

2.popen()函數
該函數打開一個指向進程的管道,該進程由派生給定的 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秒鍾。(等不等帶1秒沒有驗證,我感覺不用吧)

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

$fp = fsockopen("www.34ways.com", 80, $errno, $errstr, 30);
if (!$fp) {
    echo "$errstr ($errno)<br />\n";
} else {
    $out = "GET /index.php  / HTTP/1.1\r\n";
    $out .= "Host: www.34ways.com\r\n";
    $out .= "Connection: Close\r\n\r\n";
  
    fwrite($fp, $out);
    /*忽略執行結果
    while (!feof($fp)) {
        echo fgets($fp, 128);
    }*/
    fclose($fp);
}

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

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

  

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

關鍵代碼如下:

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

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

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

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

  

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

改進后的流程就是:

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

  

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

write.php,執行耗時 1 秒

<?php 
 
function asyn_sendmail() {
    $fp=fsockopen('localhost',80,&$errno,&$errstr,5);
    if(!$fp){
        echo "$errstr ($errno)<br />\n";
    }
    sleep(1);
    fputs($fp,"GET /sendmail.php?param=1\r\n"); #請求的資源 URL 一定要寫對
    fclose($fp);
} 
 
echo time().'<br>';
echo 'call asyn_sendmail<br>';
asyn_sendmail();
echo time().'<br>';
?>
sendmail.php,執行耗時 10<?php
//sendmail();
//sleep 10 seconds
sleep(10);
fopen('C:\'.time(),'w');
?>

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

1272472697 call asyn_sendmail
1272472698

並且在 C:\ 生成文件:

1272472708

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

 

 

異步的例子 : https://www.cnblogs.com/fps2tao/p/11562171.html

 


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM