PHP 错误自定义 set error handler函数

本文介绍PHP中的set_error_handler函数,展示如何通过自定义错误处理函数来增强安全性、改善用户体验并便于调试。文章通过实例说明如何实现错误类型的过滤、记录及定制化的错误信息展示。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

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

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的使用方法如下:

复制代码代码如下:

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

个人做了一个小测试

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
<?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,否则无效。 

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
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 ;
   }
}

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

?
1
2
3
4
// 应用到类
set_error_handler( array (& $this , "appError" ));
//示例的做法
set_error_handler( "my_error_handler" );

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

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

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

Php代码

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
class CallbackClass {
  function CallbackFunction() {
  // refers to $this
  }
  
function StaticFunction() {
  // doesn't refer to $this
  }
  }
  
function NonClassFunction( $errno , $errstr , $errfile , $errline ) {
  }
  // 三种方法如下:
  
  set_error_handler(‘NonClassFunction'); // 直接转到一个普通的函数 NonClassFunction
  set_error_handler( array (‘CallbackClass ', ‘StaticFunction' )); // 转到 CallbackClass 类下的静方法 StaticFunction
  $o =& new CallbackClass();
  set_error_handler( array ( $o , ‘CallbackFunction')); // 转到类的构造函数,其实本质上跟下面的第四条一样。
. $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
  }
  }
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值