php中set_error_handler的用法總結

來源:互聯網
上載者:User


set_error_handler()

PHP從4.1.0開始提供了自訂錯誤處理控制代碼的功能函數set_error_handler(),但很少數指令碼編寫者知道。set_error_handler這個函數可以很好地防止錯誤路徑泄露,當然還有其它更多的作用。

1.可以用來屏蔽錯誤。 出現錯誤一來會把一些資訊暴漏給使用者,極有可能成為駭客攻擊你網站的工具。 二來讓使用者覺得你的水平很挫。
2.可以記下錯誤的資訊, 及時發現一些生產環境的出現的問題。
3.可以做相應的處理, 出錯的時候可以顯示跳轉到預先定義好的出錯頁面,提供更好的使用者體驗。
4.可以作為調試工具, 一些時候必須在生產環境調試一些東西, 但又不想影響正在使用的使用者。
5.。。。。
set_error_handler的使用方法如下:

view sourceprint?1 string set_error_handler ( callback error_handler [, int error_types])

 

我們利用error_reporting();看到的錯誤資訊包括三個部分,錯誤資訊,錯誤檔案的絕對位址,錯誤出現的行數。其實還有一個是錯誤類型。Array ( [type] => 1 [message] => Call to undefined method SomeClass::somemedthod() [file] => /home/zhangy/www/aaaa/stasdf.php [line] => 67 ),頁面的絕對路徑最好不要暴露給別人,不然給有些人可稱之機,為了杜絕這一點,好多人都會採用,ini_set("display_errors",0);直接把錯誤資訊給屏蔽掉了。這樣就不方便了,如果我們要看資訊怎麼辦呢?每次查看的時候,是不是都要改一下代碼,或者是改一下apache的配置,在重起一下呢?

php有函數set_error_handler可以解決這個問題

用法如下:

mixed set_error_handler ( callback $error_handler [, int $error_types = E_ALL | E_STRICT ] )

php函數register_shutdown_function也可以解決這個問題

用法如下:

int register_shutdown_function ( string $func )

個人覺得報錯函數自己定義,至少有三點好處,

1,不會把檔案的絕對路徑顯示出來,安全些

2,即使真的出現了錯誤資訊,我們可以對錯誤資訊進行處理,讓使用者也看不到fatal error這樣的東西。使用者體驗要好

3,項目上線後,有的時候,你還是要幫使用者去解決問題,這個時候難免要去修改代碼,但是我們又要讓錯誤資訊報出來,又不能讓使用者看到,這個時候,用set_error_handler這樣的函數就很爽了。

個人做了一個小測試

<?php
error_reporting(0);

register_shutdown_function('error_alert');
function error_alert()
{
 if(is_null($e = error_get_last()) === false)
 {
 set_error_handler('errorHandler');
 if($e['type'] == 1){
 trigger_error("fatal error", E_USER_ERROR);
 }elseif($e['type'] == 8){
 trigger_error("notice", E_USER_NOTICE);
 }elseif($e['type'] == 2){
 trigger_error("warning", E_USER_WARNING);
 }else{
 trigger_error("other", E_USER_OTHER);
 }

 }else{
 echo "no error";
 }
}

 set_error_handler('errorHandler');

function errorHandler($errno, $errstr, $errfile, $errline,$errcontext)
{
 switch ($errno) {
 case E_USER_ERROR:
 echo "<b>My ERROR</b> [$errno] $errstr<br />\n";
 echo "  Fatal error on line $errline in file $errfile";
 echo ", PHP " . PHP_VERSION . " (" . PHP_OS . ")<br />\n";
 break;

 case E_USER_WARNING:
 echo "<b>My WARNING</b> [$errno] $errstr<br />\n";
 echo "  warning on line $errline in file $errfile";
 echo ", PHP " . PHP_VERSION . " (" . PHP_OS . ")<br />\n";
 break;

 case E_USER_NOTICE:
 echo "<b>My NOTICE</b> [$errno] $errstr<br />\n";
 echo "  notice on line $errline in file $errfile";
 echo ", PHP " . PHP_VERSION . " (" . PHP_OS . ")<br />\n";
 break;

 default:
 echo "Unknown error type: [$errno] $errstr<br />\n";
 echo "  warning on line $errline in file $errfile";
 echo ", PHP " . PHP_VERSION . " (" . PHP_OS . ")<br />\n";
 break;
 }

 return true;
}

class SomeClass {
 public function someMethod() {

 }
}

SomeClass::someMedthod();

$a="asdf";
foreach($a as $d){
 echo $d;
}
?>

現在我們就用自訂的錯誤處理把實際路徑過濾掉。假設有一個變數$admin,我們是用來判斷訪問者是否是管理員的(可以通過IP或者登入的使用者id來做這個判斷)

//admin為管理員的身份判定,true為管理員。 
//自訂的錯誤處理函數一定要有這4個輸入變數$errno,$errstr,$errfile,$errline,否則無效。 
function my_error_handler($errno,$errstr,$errfile,$errline) 

    //如果不是管理員就過濾實際路徑 
    if(!admin) 
    { 
        $errfile=str_replace(getcwd(),"",$errfile); 
        $errstr=str_replace(getcwd(),"",$errstr); 
    } 
    switch($errno) 
    { 
        case E_ERROR: 
        echo "ERROR: [ID $errno] $errstr (Line: $errline of $errfile) \n"; 
        echo "程式已經停止運行,請聯絡管理員。"; 
        //遇到Error級錯誤時退出指令碼 
        exit; 
        break; 
 
        case E_WARNING: 
        echo "WARNING: [ID $errno] $errstr (Line: $errline of $errfile) \n"; 
        break; 
 
        default: 
        //不顯示Notice級的錯誤 
        break; 
    } 

這樣就自訂了一個錯誤處理函數,那麼怎麼把錯誤的處理交給這個自訂函數呢?

// 應用到類 
set_error_handler(array(&$this,"appError")); 
 
//樣本的做法 
set_error_handler("my_error_handler"); 


so easy,這樣,就可以很好地解決安全和調試方便的矛盾了。而且你還可以花點心思,使錯誤提示更加美觀以配合網站的風格。

 

上面的例子中,我把錯誤資訊關掉了,而用自己的函數處理錯誤,上面的這個頁面會報fatal error,報出來的錯誤資訊我們是可以利用errorHandler來控制和處理。

好了,總結一下,下面是 set_error_handler 三種用法:

Php代碼
class CallbackClass {
function CallbackFunction() {
// refers to $this
}

function StaticFunction() {
// doesn’t refer to $this
}
}

function NonClassFunction($errno, $errstr, $errfile, $errline) {
}

// 三種方法如下:

1: set_error_handler(‘NonClassFunction’); // 直接轉到一個普通的函數 NonClassFunction

2: set_error_handler(array(‘CallbackClass’, ‘StaticFunction’)); // 轉到 CallbackClass 類下的靜方法 StaticFunction

3: $o =& new CallbackClass();
set_error_handler(array($o, ‘CallbackFunction’)); // 轉到類的建構函式,其實本質上跟下面的第四條一樣。

4. $o = new CallbackClass();

// The following may also prove useful:

class CallbackClass {
function CallbackClass() {
set_error_handler(array(&$this, ‘CallbackFunction’)); // the & is important
}

function CallbackFunction() {
// refers to $this
}
}

相關文章

聯繫我們

該頁面正文內容均來源於網絡整理,並不代表阿里雲官方的觀點,該頁面所提到的產品和服務也與阿里云無關,如果該頁面內容對您造成了困擾,歡迎寫郵件給我們,收到郵件我們將在5個工作日內處理。

如果您發現本社區中有涉嫌抄襲的內容,歡迎發送郵件至: info-contact@alibabacloud.com 進行舉報並提供相關證據,工作人員會在 5 個工作天內聯絡您,一經查實,本站將立刻刪除涉嫌侵權內容。

A Free Trial That Lets You Build Big!

Start building with 50+ products and up to 12 months usage for Elastic Compute Service

  • Sales Support

    1 on 1 presale consultation

  • After-Sales Support

    24/7 Technical Support 6 Free Tickets per Quarter Faster Response

  • Alibaba Cloud offers highly flexible support services tailored to meet your exact needs.