当前位置:首页 > PHP教程 > php函数 > 列表

php中set_error_handler的用法总结

发布:smiling 来源: PHP粉丝网  添加日期:2016-08-25 11:42:08 浏览: 评论:0 

set_error_handler() 函数设置用户自定义的错误处理函数,该函数用于创建运行时期间的用户自己的错误处理方法,该函数会返回旧的错误处理程序,若失败,则返回 null。下面来看一些例子.

set_error_handler()

PHP从4.1.0开始提供了自定义错误处理句柄的功能函数set_error_handler(),但很少数脚本编写者知道,set_error_handler这个函数可以很好地防止错误路径泄露,当然还有其它更多的作用。

1.可以用来屏蔽错误,出现错误一来会把一些信息暴漏给用户,极有可能成为黑客攻击你网站的工具,二来让用户觉得你的水平很挫。

2.可以记下错误的信息,及时发现一些生产环境的出现的问题.

3.可以做相应的处理,出错的时候可以显示跳转到预先定义好的出错页面,提供更好的用户体验。 

4.可以作为调试工具,一些时候必须在生产环境调试一些东西,但又不想影响正在使用的用户。

5.。。。。

set_error_handler的使用方法如下:

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这样的函数就很爽了。

个人做了一个小测试

  1. <?php 
  2. error_reporting(0); 
  3. register_shutdown_function('error_alert'); 
  4. function error_alert() 
  5.  if(is_null($e = error_get_last()) === false) 
  6.  { 
  7.  set_error_handler('errorHandler'); 
  8.  if($e['type'] == 1){ 
  9.  trigger_error("fatal error", E_USER_ERROR); 
  10.  }elseif($e['type'] == 8){ 
  11.  trigger_error("notice", E_USER_NOTICE); 
  12.  }elseif($e['type'] == 2){ 
  13.  trigger_error("warning", E_USER_WARNING); 
  14.  }else
  15.  trigger_error("other", E_USER_OTHER); 
  16.  } 
  17.  }else
  18.  echo "no error"
  19.  } 
  20.  set_error_handler('errorHandler'); 
  21. function errorHandler($errno$errstr$errfile$errline,$errcontext
  22.  switch ($errno) { 
  23.  case E_USER_ERROR: 
  24.  echo "<b>My ERROR</b> [$errno] $errstr<br />\n"
  25.  echo "  Fatal error on line $errline in file $errfile"
  26.  echo ", PHP " . PHP_VERSION . " (" . PHP_OS . ")<br />\n"
  27.  break
  28.  case E_USER_WARNING: 
  29.  echo "<b>My WARNING</b> [$errno] $errstr<br />\n"
  30.  echo "  warning on line $errline in file $errfile"
  31.  echo ", PHP " . PHP_VERSION . " (" . PHP_OS . ")<br />\n"
  32.  break
  33.  case E_USER_NOTICE: 
  34.  echo "<b>My NOTICE</b> [$errno] $errstr<br />\n"
  35.  echo "  notice on line $errline in file $errfile"
  36.  echo ", PHP " . PHP_VERSION . " (" . PHP_OS . ")<br />\n"
  37.  break
  38.  default
  39.  echo "Unknown error type: [$errno] $errstr<br />\n"
  40.  echo "  warning on line $errline in file $errfile"
  41.  echo ", PHP " . PHP_VERSION . " (" . PHP_OS . ")<br />\n"
  42.  break
  43.  } 
  44.  return true; 
  45. class SomeClass { 
  46.  public function someMethod() { 
  47.  } //phpfensi.com 
  48. SomeClass::someMedthod(); 
  49. $a="asdf"
  50. foreach($a as $d){ 
  51.  echo $d
  52. ?> 

现在我们就用自定义的错误处理把实际路径过滤掉,假设有一个变量$admin,我们是用来判断访问者是否是管理员的(可以通过IP或者登录的用户id来做这个判断).

admin为管理员的身份判定,true为管理员.

自定义的错误处理函数一定要有这4个输入变量$errno,$errstr,$errfile,$errline,否则无效.

  1. function my_error_handler($errno,$errstr,$errfile,$errline)   
  2. {   
  3.     //如果不是管理员就过滤实际路径   
  4.     if(!admin)   
  5.     {   
  6.         $errfile=str_replace(getcwd(),"",$errfile);   
  7.         $errstr=str_replace(getcwd(),"",$errstr);   
  8.     }   
  9.     switch($errno)   
  10.     {   
  11.         case E_ERROR:   
  12.         echo "ERROR: [ID $errno] $errstr (Line: $errline of $errfile) \n";   
  13.         echo "程序已经停止运行,请联系管理员。";   
  14.         //遇到Error级错误时退出脚本   
  15.         exit//phpfensi.com 
  16.         break;   
  17.    
  18.         case E_WARNING:   
  19.         echo "WARNING: [ID $errno] $errstr (Line: $errline of $errfile) \n";   
  20.         break;   
  21.    
  22.         default:   
  23.         //不显示Notice级的错误   
  24.         break;   
  25.     }   
  26. }  

这样就自定义了一个错误处理函数,那么怎么把错误的处理交给这个自定义函数呢?

应用到类  

set_error_handler(array(&$this,"appError"));  

示例的做法  

set_error_handler("my_error_handler"); 

so easy,这样,就可以很好地解决安全和调试方便的矛盾了,而且你还可以花点心思,使错误提示更加美观以配合网站的风格.

上面的例子中,我把错误信息关掉了,而用自己的函数处理错误,上面的这个页面会报fatal error,报出来的错误信息我们是可以利用errorHandler来控制和处理.

好了,总结一下,下面是 set_error_handler 三种用法:

  1. class CallbackClass { 
  2. function CallbackFunction() { 
  3. // refers to $this 
  4. function StaticFunction() { 
  5. // doesn’t refer to $this 
  6. 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();

  1. // The following may also prove useful: 
  2. class CallbackClass { 
  3. function CallbackClass() { 
  4. set_error_handler(array(&$this, ‘CallbackFunction’)); // the & is important 
  5. function CallbackFunction() { 
  6. // refers to $this 

Tags: set_error_handler php error

分享到: