深入PHP異步執(zhí)行的詳解
更新時(shí)間:2013年06月03日 15:00:22 作者:
本篇文章是對(duì)PHP的異步執(zhí)行進(jìn)行了詳細(xì)的分析介紹,需要的朋友參考下
Web服務(wù)器執(zhí)行一個(gè)PHP腳本,有時(shí)耗時(shí)很長(zhǎng)才能返回執(zhí)行結(jié)果,后面的腳本需要等待很長(zhǎng)一段時(shí)間才能繼續(xù)執(zhí)行。如果想實(shí)現(xiàn)只簡(jiǎn)單觸發(fā)耗時(shí)腳本的執(zhí)行而不等待執(zhí)行結(jié)果就直接執(zhí)行下一步操作,可以通過(guò)fscokopen函數(shù)來(lái)實(shí)現(xiàn)。
PHP支持socket編程,fscokopen函數(shù)返回一個(gè)到遠(yuǎn)程主機(jī)連接的句柄,可以像使用fopen返回的句柄一樣,對(duì)它進(jìn)行fwrite、fgets、fread等操作。使用fsockopen連接到本地服務(wù)器,觸發(fā)腳本執(zhí)行,然后立即返回,不等待腳本執(zhí)行完成,即可實(shí)現(xiàn)異步執(zhí)行PHP的效果。
示例代碼如下:
<?
function triggerRequest($url, $post_data = array(), $cookie = array()){
$method = "GET"; //通過(guò)POST或者GET傳遞一些參數(shù)給要觸發(fā)的腳本
$url_array = parse_url($url); //獲取URL信息
$port = isset($url_array['port'])? $url_array['port'] : 80;
$fp = fsockopen($url_array['host'], $port, $errno, $errstr, 30);
if (!$fp) {
return FALSE;
}
$getPath = $url_array['path'] ."?". $url_array['query'];
if(!empty($post_data)){
$method = "POST";
}
$header = $method . " " . $getPath;
$header .= " HTTP/1.1\r\n";
$header .= "Host: ". $url_array['host'] . "\r\n "; //HTTP 1.1 Host域不能省略
/*以下頭信息域可以省略
$header .= "User-Agent: Mozilla/5.0 (Windows; U; Windows NT 5.1; en-US; rv:1.8.1.13) Gecko/20080311 Firefox/2.0.0.13 \r\n";
$header .= "Accept: text/xml,application/xml,application/xhtml+xml,text/html;q=0.9,text/plain;q=0.8,image/png,q=0.5 \r\n";
$header .= "Accept-Language: en-us,en;q=0.5 ";
$header .= "Accept-Encoding: gzip,deflate\r\n";
*/
$header .= "Connection:Close\r\n";
if(!empty($cookie)){
$_cookie = strval(NULL);
foreach($cookie as $k => $v){
$_cookie .= $k."=".$v."; ";
}
$cookie_str = "Cookie: " . base64_encode($_cookie) ." \r\n"; //傳遞Cookie
$header .= $cookie_str;
}
if(!empty($post_data)){
$_post = strval(NULL);
foreach($post_data as $k => $v){
$_post .= $k."=".$v."&";
}
$post_str = "Content-Type: application/x-www-form-urlencoded\r\n";
$post_str .= "Content-Length: ". strlen($_post) ." \r\n"; //POST數(shù)據(jù)的長(zhǎng)度
$post_str .= $_post."\r\n\r\n "; //傳遞POST數(shù)據(jù)
$header .= $post_str;
}
fwrite($fp, $header);
//echo fread($fp, 1024); //服務(wù)器返回
fclose($fp);
return true;
}
這樣就可以通過(guò)fsockopen()函數(shù)來(lái)觸發(fā)一個(gè)PHP腳本的執(zhí)行,然后函數(shù)就會(huì)返回。 接著執(zhí)行下一步操作了。
現(xiàn)在存在一個(gè)問(wèn)題:當(dāng)客戶端斷開連接后,也就是triggerRequest發(fā)送請(qǐng)求后,立即關(guān)閉了連接,那么可能會(huì)引起服務(wù)器端正在執(zhí)行的腳本退出。
在 PHP 內(nèi)部,系統(tǒng)維護(hù)著連接狀態(tài),其狀態(tài)有三種可能的情況:
* 0 – NORMAL(正常)
* 1 – ABORTED(異常退出)
* 2 – TIMEOUT(超時(shí))
當(dāng) PHP 腳本正常地運(yùn)行 NORMAL 狀態(tài)時(shí),連接為有效。當(dāng)客戶端中斷連接時(shí),ABORTED 狀態(tài)的標(biāo)記將會(huì)被打開。遠(yuǎn)程客戶端連接的中斷通常是由用戶點(diǎn)擊 STOP 按鈕導(dǎo)致的。當(dāng)連接時(shí)間超過(guò) PHP 的時(shí)限(參閱 set_time_limit() 函數(shù))時(shí),TIMEOUT 狀態(tài)的標(biāo)記將被打開。
可以決定腳本是否需要在客戶端中斷連接時(shí)退出。有時(shí)候讓腳本完整地運(yùn)行會(huì)帶來(lái)很多方便,即使沒(méi)有遠(yuǎn)程瀏覽器接受腳本的輸出。默認(rèn)的情況是當(dāng)遠(yuǎn)程客戶端連接 中斷時(shí)腳本將會(huì)退出。該處理過(guò)程可由 php.ini 的 ignore_user_abort 或由 Apache .conf 設(shè)置中對(duì)應(yīng)的"php_value ignore_user_abort"以及 ignore_user_abort() 函數(shù)來(lái)控制。如果沒(méi)有告訴 PHP 忽略用戶的中斷,腳本將會(huì)被中斷,除非通過(guò) register_shutdown_function() 設(shè)置了關(guān)閉觸發(fā)函數(shù)。通過(guò)該關(guān)閉觸發(fā)函數(shù),當(dāng)遠(yuǎn)程用戶點(diǎn)擊 STOP 按鈕后,腳本再次嘗試輸出數(shù)據(jù)時(shí),PHP 將會(huì)檢測(cè)到連接已被中斷,并調(diào)用關(guān)閉觸發(fā)函數(shù)。
腳本也有可能被內(nèi)置的腳本計(jì)時(shí)器中斷。默認(rèn)的超時(shí)限制為 30 秒。這個(gè)值可以通過(guò)設(shè)置 php.ini 的 max_execution_time 或 Apache .conf 設(shè)置中對(duì)應(yīng)的"php_value max_execution_time"參數(shù)或者 set_time_limit() 函數(shù)來(lái)更改。當(dāng)計(jì)數(shù)器超時(shí)的時(shí)候,腳本將會(huì)類似于以上連接中斷的情況退出,先前被注冊(cè)過(guò)的關(guān)閉觸發(fā)函數(shù)也將在這時(shí)被執(zhí)行。在該關(guān)閉觸發(fā)函數(shù)中,可以通過(guò)調(diào)用 connection_status() 函數(shù)來(lái)檢查超時(shí)是否導(dǎo)致關(guān)閉觸發(fā)函數(shù)被調(diào)用。如果超時(shí)導(dǎo)致了關(guān)閉觸發(fā)函數(shù)的調(diào)用,該函數(shù)將返回 2。
需要注意的一點(diǎn)是 ABORTED 和 TIMEOUT 狀態(tài)可以同時(shí)有效。這在告訴 PHP 忽略用戶的退出操作時(shí)是可能的。PHP 將仍然注意用戶已經(jīng)中斷了連接但腳本仍然在運(yùn)行的情況。如果到了運(yùn)行的時(shí)間限制,腳本將被退出,設(shè)置過(guò)的關(guān)閉觸發(fā)函數(shù)也將被執(zhí)行。在這時(shí)會(huì)發(fā)現(xiàn)函數(shù) connection_status() 返回 3。
所以還在要觸發(fā)的腳本中指明:
<?
ignore_user_abort(TRUE);//如果客戶端斷開連接,不會(huì)引起腳本abort
set_time_limit(0);//取消腳本執(zhí)行延時(shí)上限
或使用:
<?
register_shutdown_function(callback fuction[, parameters]);//注冊(cè)腳本退出時(shí)執(zhí)行的函數(shù)
PHP支持socket編程,fscokopen函數(shù)返回一個(gè)到遠(yuǎn)程主機(jī)連接的句柄,可以像使用fopen返回的句柄一樣,對(duì)它進(jìn)行fwrite、fgets、fread等操作。使用fsockopen連接到本地服務(wù)器,觸發(fā)腳本執(zhí)行,然后立即返回,不等待腳本執(zhí)行完成,即可實(shí)現(xiàn)異步執(zhí)行PHP的效果。
示例代碼如下:
復(fù)制代碼 代碼如下:
<?
function triggerRequest($url, $post_data = array(), $cookie = array()){
$method = "GET"; //通過(guò)POST或者GET傳遞一些參數(shù)給要觸發(fā)的腳本
$url_array = parse_url($url); //獲取URL信息
$port = isset($url_array['port'])? $url_array['port'] : 80;
$fp = fsockopen($url_array['host'], $port, $errno, $errstr, 30);
if (!$fp) {
return FALSE;
}
$getPath = $url_array['path'] ."?". $url_array['query'];
if(!empty($post_data)){
$method = "POST";
}
$header = $method . " " . $getPath;
$header .= " HTTP/1.1\r\n";
$header .= "Host: ". $url_array['host'] . "\r\n "; //HTTP 1.1 Host域不能省略
/*以下頭信息域可以省略
$header .= "User-Agent: Mozilla/5.0 (Windows; U; Windows NT 5.1; en-US; rv:1.8.1.13) Gecko/20080311 Firefox/2.0.0.13 \r\n";
$header .= "Accept: text/xml,application/xml,application/xhtml+xml,text/html;q=0.9,text/plain;q=0.8,image/png,q=0.5 \r\n";
$header .= "Accept-Language: en-us,en;q=0.5 ";
$header .= "Accept-Encoding: gzip,deflate\r\n";
*/
$header .= "Connection:Close\r\n";
if(!empty($cookie)){
$_cookie = strval(NULL);
foreach($cookie as $k => $v){
$_cookie .= $k."=".$v."; ";
}
$cookie_str = "Cookie: " . base64_encode($_cookie) ." \r\n"; //傳遞Cookie
$header .= $cookie_str;
}
if(!empty($post_data)){
$_post = strval(NULL);
foreach($post_data as $k => $v){
$_post .= $k."=".$v."&";
}
$post_str = "Content-Type: application/x-www-form-urlencoded\r\n";
$post_str .= "Content-Length: ". strlen($_post) ." \r\n"; //POST數(shù)據(jù)的長(zhǎng)度
$post_str .= $_post."\r\n\r\n "; //傳遞POST數(shù)據(jù)
$header .= $post_str;
}
fwrite($fp, $header);
//echo fread($fp, 1024); //服務(wù)器返回
fclose($fp);
return true;
}
這樣就可以通過(guò)fsockopen()函數(shù)來(lái)觸發(fā)一個(gè)PHP腳本的執(zhí)行,然后函數(shù)就會(huì)返回。 接著執(zhí)行下一步操作了。
現(xiàn)在存在一個(gè)問(wèn)題:當(dāng)客戶端斷開連接后,也就是triggerRequest發(fā)送請(qǐng)求后,立即關(guān)閉了連接,那么可能會(huì)引起服務(wù)器端正在執(zhí)行的腳本退出。
在 PHP 內(nèi)部,系統(tǒng)維護(hù)著連接狀態(tài),其狀態(tài)有三種可能的情況:
* 0 – NORMAL(正常)
* 1 – ABORTED(異常退出)
* 2 – TIMEOUT(超時(shí))
當(dāng) PHP 腳本正常地運(yùn)行 NORMAL 狀態(tài)時(shí),連接為有效。當(dāng)客戶端中斷連接時(shí),ABORTED 狀態(tài)的標(biāo)記將會(huì)被打開。遠(yuǎn)程客戶端連接的中斷通常是由用戶點(diǎn)擊 STOP 按鈕導(dǎo)致的。當(dāng)連接時(shí)間超過(guò) PHP 的時(shí)限(參閱 set_time_limit() 函數(shù))時(shí),TIMEOUT 狀態(tài)的標(biāo)記將被打開。
可以決定腳本是否需要在客戶端中斷連接時(shí)退出。有時(shí)候讓腳本完整地運(yùn)行會(huì)帶來(lái)很多方便,即使沒(méi)有遠(yuǎn)程瀏覽器接受腳本的輸出。默認(rèn)的情況是當(dāng)遠(yuǎn)程客戶端連接 中斷時(shí)腳本將會(huì)退出。該處理過(guò)程可由 php.ini 的 ignore_user_abort 或由 Apache .conf 設(shè)置中對(duì)應(yīng)的"php_value ignore_user_abort"以及 ignore_user_abort() 函數(shù)來(lái)控制。如果沒(méi)有告訴 PHP 忽略用戶的中斷,腳本將會(huì)被中斷,除非通過(guò) register_shutdown_function() 設(shè)置了關(guān)閉觸發(fā)函數(shù)。通過(guò)該關(guān)閉觸發(fā)函數(shù),當(dāng)遠(yuǎn)程用戶點(diǎn)擊 STOP 按鈕后,腳本再次嘗試輸出數(shù)據(jù)時(shí),PHP 將會(huì)檢測(cè)到連接已被中斷,并調(diào)用關(guān)閉觸發(fā)函數(shù)。
腳本也有可能被內(nèi)置的腳本計(jì)時(shí)器中斷。默認(rèn)的超時(shí)限制為 30 秒。這個(gè)值可以通過(guò)設(shè)置 php.ini 的 max_execution_time 或 Apache .conf 設(shè)置中對(duì)應(yīng)的"php_value max_execution_time"參數(shù)或者 set_time_limit() 函數(shù)來(lái)更改。當(dāng)計(jì)數(shù)器超時(shí)的時(shí)候,腳本將會(huì)類似于以上連接中斷的情況退出,先前被注冊(cè)過(guò)的關(guān)閉觸發(fā)函數(shù)也將在這時(shí)被執(zhí)行。在該關(guān)閉觸發(fā)函數(shù)中,可以通過(guò)調(diào)用 connection_status() 函數(shù)來(lái)檢查超時(shí)是否導(dǎo)致關(guān)閉觸發(fā)函數(shù)被調(diào)用。如果超時(shí)導(dǎo)致了關(guān)閉觸發(fā)函數(shù)的調(diào)用,該函數(shù)將返回 2。
需要注意的一點(diǎn)是 ABORTED 和 TIMEOUT 狀態(tài)可以同時(shí)有效。這在告訴 PHP 忽略用戶的退出操作時(shí)是可能的。PHP 將仍然注意用戶已經(jīng)中斷了連接但腳本仍然在運(yùn)行的情況。如果到了運(yùn)行的時(shí)間限制,腳本將被退出,設(shè)置過(guò)的關(guān)閉觸發(fā)函數(shù)也將被執(zhí)行。在這時(shí)會(huì)發(fā)現(xiàn)函數(shù) connection_status() 返回 3。
所以還在要觸發(fā)的腳本中指明:
復(fù)制代碼 代碼如下:
<?
ignore_user_abort(TRUE);//如果客戶端斷開連接,不會(huì)引起腳本abort
set_time_limit(0);//取消腳本執(zhí)行延時(shí)上限
或使用:
<?
register_shutdown_function(callback fuction[, parameters]);//注冊(cè)腳本退出時(shí)執(zhí)行的函數(shù)
您可能感興趣的文章:
- gearman管理工具GearmanManager的安裝與php使用方法示例
- php使用gearman進(jìn)行任務(wù)分發(fā)操作實(shí)例詳解
- PHP并發(fā)多進(jìn)程處理利器Gearman使用介紹
- gearman + mysql方式實(shí)現(xiàn)持久化操作示例
- python基于json文件實(shí)現(xiàn)的gearman任務(wù)自動(dòng)重啟代碼實(shí)例
- Gearman::XS在Centos下的編譯安裝方法
- gearman隊(duì)列持久化引發(fā)的問(wèn)題及解決方法
- gearman的安裝啟動(dòng)及python API使用實(shí)例
- rhel5.7下安裝gearmand及啟動(dòng)的方法
- 實(shí)現(xiàn)PHP多線程異步請(qǐng)求的3種方法
- PHP使用gearman進(jìn)行異步的郵件或短信發(fā)送操作詳解
相關(guān)文章
解決php 處理 form 表單提交多個(gè) name 屬性值相同的 input 標(biāo)簽問(wèn)題
這篇文章主要介紹了php 處理 form 表單提交多個(gè) name 屬性值相同的 input 標(biāo)簽問(wèn)題的解決方法,需要的朋友參考下吧2017-05-05PHP中的魔術(shù)方法總結(jié)和使用實(shí)例
這篇文章主要介紹了PHP中的魔術(shù)方法總結(jié)和使用實(shí)例,魔術(shù)方法是PHP面向?qū)ο笾刑赜械奶匦?它們?cè)谔囟ǖ那闆r下被觸發(fā),都是以雙下劃線開頭,你可以把它們理解為鉤子,需要的朋友可以參考下2015-05-05PHP實(shí)現(xiàn)自動(dòng)發(fā)送郵件功能代碼(qq 郵箱)
本文給大家分享以qq郵箱為例給大家介紹PHP實(shí)現(xiàn)自動(dòng)發(fā)送郵件功能代碼,感興趣的朋友參考下吧2017-08-08PHP中cookie和session的區(qū)別實(shí)例分析
這篇文章主要介紹了PHP中cookie和session的區(qū)別,比較詳盡的分析了二者從創(chuàng)建、運(yùn)用到清除的各個(gè)流程的注意事項(xiàng),需要的朋友可以參考下2014-08-08php for 循環(huán)使用的簡(jiǎn)單實(shí)例
下面小編就為大家?guī)?lái)一篇php for 循環(huán)使用的簡(jiǎn)單實(shí)例。小編覺(jué)得挺不錯(cuò)的,現(xiàn)在就分享給大家,也給大家做個(gè)參考。一起跟隨小編過(guò)來(lái)看看吧2016-06-06thinkphp3.2中實(shí)現(xiàn)phpexcel導(dǎo)出帶生成圖片示例
本篇文章主要介紹了thinkphp3.2中實(shí)現(xiàn)phpexcel導(dǎo)出帶生成圖片示例,小編覺(jué)得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過(guò)來(lái)看看吧2017-02-02