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