浅谈php的错误处理
一直对php的错误处理有些不理解,最近接触的比较多,仔细想来,慢慢就有了豁然开朗的感觉。
----------------------------------------------------------------------------------------------------------
先来谈谈php中的 error_reporting() 这个函数:
这里首先,必须说一下这个函数的参数。如下所示:
值 | 常量 | 说明 | 备注 |
---|---|---|---|
1 | E_ERROR (integer) |
致命的运行时错误。这类错误一般是不可恢复的情况,例如内存分配导致的问题。后果是导致脚本终止不再继续运行。 | |
2 | E_WARNING (integer) |
运行时警告 (非致命错误)。仅给出提示信息,但是脚本不会终止运行。 | |
4 | E_PARSE (integer) |
编译时语法解析错误。解析错误仅仅由分析器产生。 | |
8 | E_NOTICE (integer) |
运行时通知。表示脚本遇到可能会表现为错误的情况,但是在可以正常运行的脚本里面也可能会有类似的通知。 | |
16 | E_CORE_ERROR (integer) |
在PHP初始化启动过程中发生的致命错误。该错误类似 E_ERROR ,但是是由PHP引擎核心产生的。 |
since PHP 4 |
32 | E_CORE_WARNING (integer) |
PHP初始化启动过程中发生的警告 (非致命错误) 。类似 E_WARNING ,但是是由PHP引擎核心产生的。 |
since PHP 4 |
64 | E_COMPILE_ERROR (integer) |
致命编译时错误。类似E_ERROR , 但是是由Zend脚本引擎产生的。 |
since PHP 4 |
128 | E_COMPILE_WARNING (integer) |
编译时警告 (非致命错误)。类似 E_WARNING ,但是是由Zend脚本引擎产生的。 |
since PHP 4 |
256 | E_USER_ERROR (integer) |
用户产生的错误信息。类似 E_ERROR , 但是是由用户自己在代码中使用PHP函数 trigger_error()来产生的。 |
since PHP 4 |
512 | E_USER_WARNING (integer) |
用户产生的警告信息。类似 E_WARNING , 但是是由用户自己在代码中使用PHP函数 trigger_error()来产生的。 |
since PHP 4 |
1024 | E_USER_NOTICE (integer) |
用户产生的通知信息。类似 E_NOTICE , 但是是由用户自己在代码中使用PHP函数 trigger_error()来产生的。 |
since PHP 4 |
2048 | E_STRICT (integer) |
启用 PHP 对代码的修改建议,以确保代码具有最佳的互操作性和向前兼容性。 | since PHP 5 |
4096 | E_RECOVERABLE_ERROR (integer) |
可被捕捉的致命错误。 它表示发生了一个可能非常危险的错误,但是还没有导致PHP引擎处于不稳定的状态。 如果该错误没有被用户自定义句柄捕获 (参见 set_error_handler()),将成为一个 E_ERROR 从而脚本会终止运行。 |
since PHP 5.2.0 |
8192 | E_DEPRECATED (integer) |
运行时通知。启用后将会对在未来版本中可能无法正常工作的代码给出警告。 | since PHP 5.3.0 |
16384 | E_USER_DEPRECATED (integer) |
用户产少的警告信息。 类似 E_DEPRECATED , 但是是由用户自己在代码中使用PHP函数 trigger_error()来产生的。 |
since PHP 5.3.0 |
30719 | E_ALL (integer) |
E_STRICT 出外的所有错误和警告信息。 |
30719 in PHP 5.3.x, 6143 in PHP 5.2.x, 2047 previously |
不难发现,所有这些可用的参数都是预定义常量且都是二进制的整型数。既然是二进制,必然会涉及到二进制的运算,尤其是 |(或)、&(与) 。
再来看手册的写法:
1 // 关闭所有PHP错误报告 2 error_reporting(0); 3 4 // Report simple running errors 5 error_reporting(E_ERROR | E_WARNING | E_PARSE); 6 7 // 报告 E_NOTICE也挺好 (报告未初始化的变量 8 // 或者捕获变量名的错误拼写) 9 error_reporting(E_ERROR | E_WARNING | E_PARSE | E_NOTICE); 10 11 // 除了 E_NOTICE,报告其他所有错误 12 // 这是在 php.ini 里的默认设置 13 error_reporting(E_ALL ^ E_NOTICE); 14 15 // 报告所有 PHP 错误 (参见 changelog) 16 error_reporting(E_ALL); 17 18 // 报告所有 PHP 错误 19 error_reporting(-1); 20 21 // 和 error_reporting(E_ALL); 一样 22 ini_set('error_reporting', E_ALL);
那么,就不难理解 E_ERROR | E_WARNING | E_PARSE | E_NOTICE 这种写法了。也就是 | 即同时报告某个级别错误的意思。
当我们查看当前报错级别时。可用 echo error_reporting();结果是一个十进制数字,再结合上面那个表,就可以理解当前的报错级别了。
当我们想设置报错级别时,可用 error_reporting(E_ALL) 等;
----------------------------------------------------------------------------------------------
再说说 trigger_error() 这个函数:
手册上已经讲的非常详细了,这个函数的作用就是触发一个用户自定义的错误。
看手册的说法:
trigger_error — 产生一个用户级别的 error/warning/notice 信息
说明
$error_msg
[, int $error_type
= E_USER_NOTICE ] )用于触发一个用户级别的错误条件,它能结合内置的错误处理器所关联,或者可以使用用户定义的函数作为新的错误处理程序( set_error_handler())。
该函数在你运行出现异常时,需要产生一个特定的响应时非常有用。
参数
error_msg
-
该 error 的特定错误信息,长度限制在了1024个字符。超过1024长度的字符都会被截断。
error_type
-
该 error 所特定的错误类型。仅 E_USER 系列常量对其有效,默认是
E_USER_NOTICE
。
返回值
如果指定了错误的 error_type
会返回 FALSE
,正确则返回 TRUE
。
----------------------------------------------------------------------------------------------------
接下来也是一个常用的函数 set_error_handler() :
同样,还是看手册:
说明
设置一个用户的函数(error_handler
)来处理脚本中出现的错误。
本函数可以用你自己定义的方式来处理运行中的错误, 例如,在应用程序中严重错误发生时,或者在特定条件下触发了一个错误(使用 trigger_error()),你需要对数据/文件做清理回收。
重要的是要记住 error_types
里指定的错误类型都会绕过 PHP 标准错误处理程序, 除非回调函数返回了 FALSE
。 error_reporting() 设置将不会起到作用而你的错误处理函数继续会被调用 —— 不过你仍然可以获取 error_reporting 的当前值,并做适当处理。 需要特别注意的是带 @ error-control operator 前缀的语句发生错误时,这个值会是 0。
同时注意,在需要时你有责任使用 die()。 如果错误处理程序返回了,脚本将会继续执行发生错误的后一行。
以下级别的错误不能由用户定义的函数来处理: E_ERROR
、 E_PARSE
、 E_CORE_ERROR
、 E_CORE_WARNING
、 E_COMPILE_ERROR
、 E_COMPILE_WARNING
,和在 调用set_error_handler() 函数所在文件中产生的大多数 E_STRICT
。
如果错误发生在脚本执行之前(比如文件上传时),将不会 调用自定义的错误处理程序因为它尚未在那时注册。
参数
error_handler
-
用户的函数需要接受两个参数:错误码和描述错误的 string。 另外有可能提供三个可选参数:发生错误的文件名、发生错误的行号 以及发生错误的上下文(一个指向错误发生时活动符号表的 array)。 该函数可以表示为:
handler ( int$errno
, string$errstr
[, string$errfile
[, int$errline
[, array$errcontext
]]] )errno
- 第一个参数
errno
,包含了错误的级别,是一个 integer。 errstr
- 第二个参数
errstr
,包含了错误的信息,是一个 string。 errfile
- 第三个参数是可选的,
errfile
, 包含了发生错误的文件名,是一个 string。 errline
- 第四个参数是一个可选项,
errline
, 包含了错误发生的行号,是一个 integer。 errcontext
- 第五个可选参数,
errcontext
, 是一个指向错误发生时活动符号表的 array。 也就是说,errcontext
会包含错误触发处作用域内所有变量的数组。 用户的错误处理程序不应该修改错误上下文(context)。
如果函数返回
FALSE
,标准错误处理处理程序将会继续调用。 error_types
-
就像error_reporting 的 ini 设置能够控制错误的显示一样, 此参数能够用于屏蔽
error_handler
的触发。 如果没有该掩码, 无论 error_reporting 是如何设置的,error_handler
都会在每个错误发生时被调用。
返回值
如果之前有定义过错误处理程序,则返回该程序名称的 string;如果是内置的错误处理程序,则返回 NULL
。 如果你指定了一个无效的回调函数,同样会返回 NULL
。 如果之前的错误处理程序是一个类的方法,此函数会返回一个带类和方法名的索引数组(indexed array)。
结合手册给出的示例,应该就比较了解这几个函数的用法了:
1 // error handler function 2 function myErrorHandler($errno, $errstr, $errfile, $errline) 3 { 4 if (!(error_reporting() & $errno)) { 5 // This error code is not included in error_reporting 6 return; 7 } 8 9 switch ($errno) { 10 case E_USER_ERROR: 11 echo "<b>My ERROR</b> [$errno] $errstr<br /><br/>"; 12 echo " Fatal error on line $errline in file $errfile"; 13 echo ", PHP " . PHP_VERSION . " (" . PHP_OS . ")<br /><br/>"; 14 echo "Aborting...<br /><br/>"; 15 exit(1); 16 break; 17 18 case E_USER_WARNING: 19 echo "<b>My WARNING</b> [$errno] $errstr<br /><br/>"; 20 break; 21 22 case E_USER_NOTICE: 23 echo "<b>My NOTICE</b> [$errno] $errstr<br /><br/>"; 24 break; 25 26 default: 27 echo "Unknown error type: [$errno] $errstr<br /><br/>"; 28 break; 29 } 30 31 /* Don't execute PHP internal error handler */ 32 return true; 33 } 34 35 // function to test the error handling 36 function scale_by_log($vect, $scale) 37 { 38 if (!is_numeric($scale) || $scale <= 0) { 39 trigger_error("log(x) for x <= 0 is undefined, you used: scale = $scale", E_USER_ERROR); 40 } 41 42 if (!is_array($vect)) { 43 trigger_error("Incorrect input vector, array of values expected", E_USER_WARNING); 44 return null; 45 } 46 47 $temp = array(); 48 foreach($vect as $pos => $value) { 49 if (!is_numeric($value)) { 50 trigger_error("Value at position $pos is not a number, using 0 (zero)", E_USER_NOTICE); 51 $value = 0; 52 } 53 $temp[$pos] = log($scale) * $value; 54 } 55 56 return $temp; 57 } 58 59 // set to the user defined error handler 60 $old_error_handler = set_error_handler("myErrorHandler"); 61 62 // trigger some errors, first define a mixed array with a non-numeric item 63 echo "vector a<br/>"; 64 $a = array(2, 3, "foo", 5.5, 43.3, 21.11); 65 print_r($a); 66 67 // now generate second array 68 echo "----<br/>vector b - a notice (b = log(PI) * a)<br/>"; 69 /* Value at position $pos is not a number, using 0 (zero) */ 70 $b = scale_by_log($a, M_PI); 71 print_r($b); 72 73 // this is trouble, we pass a string instead of an array 74 echo "----<br/>vector c - a warning<br/>"; 75 /* Incorrect input vector, array of values expected */ 76 $c = scale_by_log("not array", 2.3); 77 var_dump($c); // NULL 78 79 // this is a critical error, log of zero or negative number is undefined 80 echo "----<br/>vector d - fatal error<br/>"; 81 /* log(x) for x <= 0 is undefined, you used: scale = $scale" */ 82 $d = scale_by_log($a, -2.5); 83 var_dump($d); // Never reached
输出:
vector a Array ( [0] => 2 [1] => 3 [2] => foo [3] => 5.5 [4] => 43.3 [5] => 21.11 ) ---- vector b - a notice (b = log(PI) * a) <b>My NOTICE</b> [1024] Value at position 2 is not a number, using 0 (zero)<br /> Array ( [0] => 2.2894597716988 [1] => 3.4341896575482 [2] => 0 [3] => 6.2960143721717 [4] => 49.566804057279 [5] => 24.165247890281 ) ---- vector c - a warning <b>My WARNING</b> [512] Incorrect input vector, array of values expected<br /> NULL ---- vector d - fatal error <b>My ERROR</b> [256] log(x) for x <= 0 is undefined, you used: scale = -2.5<br /> Fatal error on line 35 in file trigger_error.php, PHP 5.2.1 (FreeBSD)<br /> Aborting...<br />