php異步調試和線上調試網(wǎng)站程序
創(chuàng)新互聯(lián)公司,專注為中小企業(yè)提供官網(wǎng)建設、營銷型網(wǎng)站制作、自適應網(wǎng)站建設、展示型成都做網(wǎng)站、成都網(wǎng)站設計等服務,幫助中小企業(yè)通過網(wǎng)站體現(xiàn)價值、有效益。幫助企業(yè)快速建站、解決網(wǎng)站建設與網(wǎng)站營銷推廣問題。
php異步調試和線上調試網(wǎng)站程序既方便網(wǎng)站程序錯誤調試,又不影響網(wǎng)站的正常運行的調試方法。下面是我為大家?guī)Я说膒hp異步調試和線上調試網(wǎng)站程序,歡迎閱讀。
php異步調試和線上調試網(wǎng)站程序
代碼如下
//ini_set('error_reporting',E_ALL ^ E_NOTICE);//顯示所有除了notice類型的錯誤信息
ini_set('error_reporting',E_ALL);//顯示所有錯誤信息
ini_set('display_errors',off);//禁止將錯誤信息輸出到輸出端
ini_set('log_errors',On);//開啟錯誤日志記錄
ini_set('error_log','C:/phpernote');//定義錯誤日志存儲位置
另外附加兩句比較常用的排除錯誤信息的PHP語句:
代碼如下
@ini_set('memory_limit','500M');//設置程序可占用最大內存為500MB
@ini_set('max_execution_time','180');//設置允許程序最長的執(zhí)行時間為180秒
補充
die()和exit()也是我們常用的php調試一個方法
die()和exit()函數(shù)都有終止線程的作用,是php斷點調試需要使用的最主要的函數(shù),它們也是php程序員使用非常頻繁的函數(shù)。然而兩者又有什么區(qū)別呢?在程序調試時需要注意什么問題呢?
die()函數(shù)一般與“or”一并使用,寫作“or die()”,經??吹竭@樣的語句:
代碼如下
$file = fopen($filename, 'r') or die("抱歉,無法打開: $filename")
or在這里是這樣理解的,因為在PHP中并不區(qū)分數(shù)據(jù)類型,所以$file既可以是int也可以bool,所以這樣的語句不會報錯。但其處理過程可能有些朋友不大明白。其實在大多數(shù)的語言中, bool or bool這樣的語句中,如果前一個值為真后一個值就不會再判斷了。這里也是的,所以如果fopen函數(shù)執(zhí)行正確的話,會返回一個大于0的int值(這其實就是"真"),后面的語句就不會執(zhí)行了。如果fopen函數(shù)執(zhí)行失敗,就會返回false,那么就會判斷后面的表達式是否為真了。結果執(zhí)行了die()之后,不管返回什么,程序都已經停止執(zhí)行了,并且顯示指定的.出錯信息,也就達到了調試的目的。就這樣。
實際上,die和exit是等價的,都是用來終止當前腳本。
php手冊對兩者的解釋如是說:
exit() 函數(shù)輸出一條消息,并退出當前腳本。該函數(shù)是 die() 函數(shù)的別名。
die() 函數(shù)輸出一條消息,并退出當前腳本。該函數(shù)是 exit() 函數(shù)的別名。
實例:
代碼如下
?php $site = ""; fopen($site,"r") or exit("Unable to connect to $site"); ?
?php $site = ""; fopen($site,"r") or die("Unable to connect to $site"); ?
var_dump()和print_r()
var_dump -- 打印變量的相關信息
void var_dump ( mixed expression [, mixed expression [, ...]] )
此函數(shù)顯示關于一個或多個表達式的結構信息,包括表達式的類型與值。數(shù)組將遞歸展開值,通過縮進顯示其結構。
提示: 為了防止程序直接將結果輸出到瀏覽器,可以使用輸出控制函數(shù)(output-control functions)來捕獲此函數(shù)的輸出,并把它們保存到一個例如 string 類型的變量中。
代碼如下
?php
$a = array (1, 2, array ("a", "b", "c"));
var_dump ($a);
$b = 3.1;
$c = TRUE;
var_dump($b,$c);
?
var_dump()可以輸出多個變量,如:var_dump($b,$c)
print_r -- 打印關于變量的易于理解的信息
bool print_r ( mixed expression [, bool return] )
注: 參數(shù) return 是在 PHP 4.3.0 的時候加上的
print_r() 顯示關于一個變量的易于理解的信息。如果給出的是 string、integer 或 float,將打印變量值本身。如果給出的是 array,將會按照一定格式顯示鍵和元素。object 與數(shù)組類似。
記住,print_r() 將把數(shù)組的指針移到最后邊。使用reset() 可讓指針回到開始處。
代碼如下
pre
?php
$a = array ('a' = 'apple',
'b' = 'banana',
'c' = array ('x','y','z'));
print_r ($a);
?
/pre
上邊的代碼將輸出:
pre Array ( [a] = apple [b] = banana [c] = Array ( [0] = x [1] = y [2] = z ) ) /pre
如果想捕捉 print_r() 的輸出,可使用 return 參數(shù)。若此參數(shù)設為 TRUE,print_r() 將不打印結果(此為默認動作),而是返回其輸出。
例子:return 參數(shù)示例
代碼如下
?php
$b = array ('m' = 'monkey',
'foo' = 'bar',
'x' = array ('x', 'y', 'z'));
$results = print_r ($b, true); //$results 包含了 print_r 的輸出結果
?
注: 如果想在 PHP 4.3.0 之前的版本中捕捉 print_r() 的輸出,可使用輸出控制函數(shù)。
注: 在 PHP 4.0.4 之前的版本中,如果給出的 array 或 object 包含了直接或間接指向自身的引用,print_r() 將永遠繼續(xù)下去。print_r($GLOBALS) 就是一個例子,因為 $GLOBALS 自身即是全局變量,其包含了指向自身的引用。
下面的幾個函數(shù)可以讓你隨時查看程序中任何變量的類型及其值。
代碼如下
function ss_array_as_string (&$array, $column = 0) {
$str = "Array(
n";
while(list($var, $val) = each($array)){
for ($i = 0; $i $column+1; $i++){
$str .= "    ";
}
$str .= $var. ==; ;
$str .= ss_as_string($val, $column+1)."
n";
}
for ($i = 0; $i $column; $i++){
$str .= "    ";
}
return $str.);
}
function ss_object_as_string (&$object, $column = 0) {
if (empty($object-;classname)) {
return "$object";
}
else {
$str = $object-;classname."(
n";
while (list(,$var) = each($object-;persistent_slots)) {
for ($i = 0; $i $column; $i++){
$str .= "    ";
}
global $$var;
$str .= $var. ==; ;
$str .= ss_as_string($$var, column+1)."
n";
}
for ($i = 0; $i $column; $i++){
$str .= "    ";
}
return $str.);
}
}
function ss_as_string (&$thing, $column = 0) {
if (is_object($thing)) {
return ss_object_as_string($thing, $column);
}
elseif (is_array($thing)) {
return ss_array_as_string($thing, $column);
}
elseif (is_double($thing)) {
return "Double(".$thing.")";
}
elseif (is_long($thing)) {
return "Long(".$thing.")";
}
elseif (is_string($thing)) {
return "String(".$thing.")";
}
else {
return "Unknown(".$thing.")";
}
}
;
Web服務器執(zhí)行一個PHP腳本,有時耗時很長才能返回執(zhí)行結果,后面的腳本需要等待很長一段時間才能繼續(xù)執(zhí)行。如果想實現(xiàn)只簡單觸發(fā)耗時腳本的執(zhí)行而不等待執(zhí)行結果就直接執(zhí)行下一步操作,可以通過fscokopen函數(shù)來實現(xiàn)。
PHP支持socket編程,fscokopen函數(shù)返回一個到遠程主機連接的句柄,可以像使用fopen返回的句柄一樣,對它進行fwrite、fgets、fread等操作。使用fsockopen連接到本地服務器,觸發(fā)腳本執(zhí)行,然后立即返回,不等待腳本執(zhí)行完成,即可實現(xiàn)異步執(zhí)行PHP的效果。
示例代碼如下:
functiontriggerRequest($url,$post_data=array(),$cookie=array()){
$method="GET";//通過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){
returnFALSE;
}
$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";//HTTP1.1Host域不能省略
/*以下頭信息域可以省略
$header.="User-Agent:Mozilla/5.0(Windows;U;
WindowsNT5.1;en-US;rv:1.8.1.13)Gecko/20080311Firefox/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($cookieas$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_dataas$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ù)的長度
$post_str.=$_post."\r\n\r\n";//傳遞POST數(shù)據(jù)
$header.=$post_str;
}
fwrite($fp,$header);
//echofread($fp,1024);//服務器返回
fclose($fp);
returntrue;
}
這樣就可以通過fsockopen()函數(shù)來觸發(fā)一個PHP腳本的執(zhí)行,然后函數(shù)就會返回。接著執(zhí)行下一步操作了。
現(xiàn)在存在一個問題:當客戶端斷開連接后,也就是triggerRequest發(fā)送請求后,立即關閉了連接,那么可能會引起服務器端正在執(zhí)行的腳本退出。
在PHP內部,系統(tǒng)維護著連接狀態(tài),其狀態(tài)有三種可能的情況:
*0–NORMAL(正常)
*1–ABORTED(異常退出)
*2–TIMEOUT(超時)
當PHP腳本正常地運行NORMAL狀態(tài)時,連接為有效。當客戶端中斷連接時,ABORTED狀態(tài)的標記將會被打開。遠程客戶端連接的中斷通常是由用戶點擊STOP按鈕導致的。當連接時間超過PHP的時限(參閱set_time_limit()函數(shù))時,TIMEOUT狀態(tài)的標記將被打開。
可以決定腳本是否需要在客戶端中斷連接時退出。有時候讓腳本完整地運行會帶來很多方便,即使沒有遠程瀏覽器接受腳本的輸出。默認的情況是當遠程客戶端連接中斷時腳本將會退出。該處理過程可由php.ini的ignore_user_abort或由Apache.conf設置中對應的"php_valueignore_user_abort"以及ignore_user_abort()函數(shù)來控制。如果沒有告訴PHP忽略用戶的中斷,腳本將會被中斷,除非通過register_shutdown_function()設置了關閉觸發(fā)函數(shù)。通過該關閉觸發(fā)函數(shù),當遠程用戶點擊STOP按鈕后,腳本再次嘗試輸出數(shù)據(jù)時,PHP將會檢測到連接已被中斷,并調用關閉觸發(fā)函數(shù)。
腳本也有可能被內置的腳本計時器中斷。默認的超時限制為30秒。這個值可以通過設置php.ini的max_execution_time或Apache.conf設置中對應的"php_valuemax_execution_time"參數(shù)或者set_time_limit()函數(shù)來更改。當計數(shù)器超時的時候,腳本將會類似于以上連接中斷的情況退出,先前被注冊過的關閉觸發(fā)函數(shù)也將在這時被執(zhí)行。在該關閉觸發(fā)函數(shù)中,可以通過調用connection_status()函數(shù)來檢查超時是否導致關閉觸發(fā)函數(shù)被調用。如果超時導致了關閉觸發(fā)函數(shù)的調用,該函數(shù)將返回2。
需要注意的一點是ABORTED和TIMEOUT狀態(tài)可以同時有效。這在告訴PHP忽略用戶的退出操作時是可能的。PHP將仍然注意用戶已經中斷了連接但腳本仍然在運行的情況。如果到了運行的時間限制,腳本將被退出,設置過的關閉觸發(fā)函數(shù)也將被執(zhí)行。在這時會發(fā)現(xiàn)函數(shù)connection_status()返回3。
所以還在要觸發(fā)的腳本中指明:
ignore_user_abort(TRUE);//如果客戶端斷開連接,不會引起腳本abort
set_time_limit(0);//取消腳本執(zhí)行延時上限
或使用:
register_shutdown_function(callbackfuction[,parameters]);//注冊腳本退出時執(zhí)行的函數(shù)
使用Ajax 與 img 標記
原理,服務器返回的html中插入Ajax 代碼或 img 標記,img的src為需要執(zhí)行的程序。
優(yōu)點:實現(xiàn)簡單,服務端無需執(zhí)行任何調用
缺點:在執(zhí)行期間,瀏覽器會一直處于loading狀態(tài),因此這種方法并不算真正的異步調用。
$.get("doRequest.php",?{?name:?"fdipzone"}?);
img?src="doRequest.php?name=fdipzone"
使用popen
使用popen執(zhí)行命令,語法:
優(yōu)點:執(zhí)行速度快
缺點:
1).只能在本機執(zhí)行
2).不能傳遞大量參數(shù)
3).訪問量高時會創(chuàng)建很多進程
//?popen?—?打開進程文件指針??
resource?popen?(?string?$command?,?string?$mode?)
pclose(popen('php?/home/fdipzone/doRequest.php?',?'r'));
使用curl
設置curl的超時時間 CURLOPT_TIMEOUT 為1 (最小為1),因此客戶端需要等待1秒
?php?
$ch?=?curl_init();?
$curl_opt?=?array(?
CURLOPT_URL,?''
CURLOPT_RETURNTRANSFER,1,?
CURLOPT_TIMEOUT,1?
);?
curl_setopt_array($ch,?$curl_opt);?
curl_exec($ch);?
curl_close($ch);?
?
使用fsockopen
fsockopen是最好的,缺點是需要自己拼接header部分。
?php?
$url?=?'';?
$param?=?array(?
'name'='fdipzone',?
'gender'='male',?
'age'=30?
);?
doRequest($url,?$param);?
function?doRequest($url,?$param=array()){?
$urlinfo?=?parse_url($url);?
$host?=?$urlinfo['host'];?
$path?=?$urlinfo['path'];?
$query?=?isset($param)??http_build_query($param)?:?'';?
$port?=?80;?
$errno?=?0;?
$errstr?=?'';?
$timeout?=?10;?
$fp?=?fsockopen($host,?$port,?$errno,?$errstr,?$timeout);?
$out?=?"POST?".$path."?HTTP/1.1\r\n";?
$out?.=?"host:".$host."\r\n";?
$out?.=?"content-length:".strlen($query)."\r\n";?
$out?.=?"content-type:application/x-www-form-urlencoded\r\n";?
$out?.=?"connection:close\r\n\r\n";?
$out?.=?$query;?
fputs($fp,?$out);?
fclose($fp);?
}?
?
注意:當執(zhí)行過程中,客戶端連接斷開或連接超時,都會有可能造成執(zhí)行不完整,因此需要加上
ignore_user_abort(true);?//?忽略客戶端斷開?
set_time_limit(0);????//?設置執(zhí)行不超時
利用php的系統(tǒng)調用,開啟新的進程來實現(xiàn)。
php
提供了fsockopen函數(shù),此函數(shù)的功能為初始化一個套接字連接到指定主機,默認情況下將以阻塞模式開啟套接字連接。當然你可以通過stream_set_blocking()將它轉換到非阻塞模式。這是關鍵。所以,思路就是:開啟一個非阻塞的套接字連接到本機,本機收到之后作一些耗時處理。
PHP是順序執(zhí)行的語言,注定無法單獨依靠PHP本身去實現(xiàn)異步執(zhí)行。
但可以借助比如在HTML中ajax的異步請求去實現(xiàn)自己想要的效果。
php語言得用fsockopen()函數(shù),實現(xiàn)腳本異步運行,代碼如下
異步請求函數(shù)(用debug參數(shù)若為true則為用為調試,開啟調試可以看到異步的執(zhí)行情況,但是失去異步的效果)
main.php
?php
/**
*?異步請求
*?@copyright??Copyright?(c)?Hangzhou?Technology?Co.,Ltd.?()
*?@author?????$Author:?juny?$
*?@version????$Id:?main.php?332?2018-09-23?09:15:08Z?juny?$
*/
function?request_by_fsockopen($url,$post_data=array(),$debug=false){
$url_array?=?parse_url($url);
$hostname?=?$url_array['host'];
$port?=?isset($url_array['port'])??$url_array['port']?:?80;
@$requestPath?=?$url_array['path']?."?".?$url_array['query'];
$fp?=?fsockopen($hostname,?$port,?$errno,?$errstr,?10);
if?(!$fp)?{
echo?"$errstr?($errno)";
return?false;
}
$method?=?"GET";
if(!empty($post_data)){
$method?=?"POST";
}
$header?=?"$method?$requestPath?HTTP/1.1\r\n";
$header.="Host:?$hostname\r\n";
if(!empty($post_data)){
$_post?=?strval(NULL);
foreach($post_data?as?$k?=?$v){
$_post[]=?$k."=".urlencode($v);//必須做url轉碼以防模擬post提交的數(shù)據(jù)中有符而導致post參數(shù)鍵值對紊亂
}
$_post?=?implode('',?$_post);
$header?.=?"Content-Type:?application/x-www-form-urlencoded\r\n";//POST數(shù)據(jù)
$header?.=?"Content-Length:?".?strlen($_post)?."\r\n";//POST數(shù)據(jù)的長度
$header.="Connection:?Close\r\n\r\n";//長連接關閉
$header?.=?$_post;?//傳遞POST數(shù)據(jù)
}else{
$header.="Connection:?Close\r\n\r\n";//長連接關閉
}
fwrite($fp,?$header);
//-----------------調試代碼區(qū)間-----------------
//注如果開啟下面的注釋,異步將不生效可是方便調試
if($debug){
$html?=?'';
while?(!feof($fp))?{
$html.=fgets($fp);
}
echo?$html;
}
//-----------------調試代碼區(qū)間-----------------
fclose($fp);
}
$data=array('name'='guoyu','pwd'='123456');
$url='';
request_by_fsockopen($url,$data,true);//
other.php
?php
header("content-type:text/html;charset=utf-8");
//error_reporting(0);
//ini_set('html_errors',false);
//ini_set('display_errors',false);
$name?=?isset($_POST['name'])?$_POST['name']:'';
$pwd?=?isset($_POST['pwd'])?$_POST['pwd']:'';
echo?$name.$pwd;
echo?'success?ok';
die;
?
使用實例:
[運行的main.php主腳本文件]
$data=array('name'='guoyu','pwd'='123456');
$url='';
request_by_fsockopen($url,$data,true);//把應用B的用戶表異步-同步數(shù)據(jù)
[導步執(zhí)行文件other.php]
在other.php中便可以用$_POST接收main.php提交過來的參數(shù),從而進行下一步操作
以上就是php如何實現(xiàn)腳本異步執(zhí)行的方法具體分析的詳細內容.