PHP的全局错误处理详解


Posted in PHP onApril 25, 2016

本文目的

PHP的全局错误处理,在开发项目的时候很有用,可以帮助开发者快速定位一些问题,提高工作效率。默认情况下,全局错误会直接输出,但是最近开发时使用的一个框架库对全局错误处理进行了设定,导致很多错误信息没有输出,在定位问题上有一定的耗时。所以,研究了一下此库的实现,发现它设定了error_reporting和set_error_handler,导致此现象。现在记录一下这两个函数的用法,作为备忘录。

背景

PHP没有类型检测,开发人员比较容易输入错误单词,引起致命错误,最终导致脚本停止执行。如果这个时候,没有得到任何错误消息,那么会是一件很痛苦的事情。你不得不从脚本的第一行代码开始调试,在成千上万行的代码中不断的print或者echo,直到定位到这个输错的单词。然后,有不得不原路返回,将先前添加的print或echo全部删除。这时一件及其枯燥乏味的工作。

一般情况

正常情况下,php会将致命错误直接输出,会将错误的出处(文件地址,行号)和原因等输出,这样,开发着可以很方便的定位到问题。

但是有些时候,可能由于php.ini的设置问题,可能是第三方框架配置的问题,导致这些信息没有输出,那么此时,必须学会自己设置相关参数,输出这些错误信息,帮助快速定位问题。

error_reporting

error_reporting是一个php的全局配置参数,在php.ini中。用于配置错误输出级别,参数是比特位,可以用来设置错误输出的级别,下面是从php.ini中copy出来的信息:

; error_reporting is a bit-field. Or each number up to get desired error
; reporting level
; E_ALL - All errors and warnings (doesn't include E_STRICT)
; E_ERROR - fatal run-time errors
; E_RECOVERABLE_ERROR - almost fatal run-time errors
; E_WARNING - run-time warnings (non-fatal errors)
; E_PARSE - compile-time parse errors
; E_NOTICE - run-time notices (these are warnings which often result
; from a bug in your code, but it's possible that it was
; intentional (e.g., using an uninitialized variable and
; relying on the fact it's automatically initialized to an
; empty string)
; E_STRICT - run-time notices, enable to have PHP suggest changes
; to your code which will ensure the best interoperability
; and forward compatibility of your code
; E_CORE_ERROR - fatal errors that occur during PHP's initial startup
; E_CORE_WARNING - warnings (non-fatal errors) that occur during PHP's
; initial startup
; E_COMPILE_ERROR - fatal compile-time errors
; E_COMPILE_WARNING - compile-time warnings (non-fatal errors)
; E_USER_ERROR - user-generated error message
; E_USER_WARNING - user-generated warning message
; E_USER_NOTICE - user-generated notice message
;
; Examples:
;
; - Show all errors, except for notices and coding standards warnings
;
;error_reporting = E_ALL & ~E_NOTICE
;
; - Show all errors, except for notices
;
;error_reporting = E_ALL & ~E_NOTICE | E_STRICT
;
; - Show only errors
;
;error_reporting = E_COMPILE_ERROR|E_RECOVERABLE_ERROR|E_ERROR|E_CORE_ERROR
;
; - Show all errors except for notices and coding standards warnings
;
error_reporting = E_ALL & ~E_NOTICE

默认情况下,php会输出所有错误信息,除了notice。同样,php标准函数中提供了名称相同的函数error_reporting(int $level),用于在php脚本中,完成同样的功能。这样将不会影响其他程序。值得注意的是,$level为0的时候是关闭错误输出,也就是任何错误都不会输出。

set_error_handler

php的默认错误处理是将消息输出。但是,有时候需要定义一些其他操作,这时就需要自定义错误处理函数。php提供内置函数set_error_handler可以帮助我们注册自己的错误处理函数。函数原型如下:

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

值得注意的是,即使注册了错误处理函数,默认的行为仍然会执行,也就是错误出现时,仍然会输出错误信息,所以需要在程序中显示的将错误级别设置为0,然后在注册自己的的错误处理函数。这种方式,在生产环境下,尤其重要,因为即时出错,敏感内部错误信息也不会暴露给潜在的恶意用户。还有很重要的一点需要指出,自定义错误处理函数不能处理fatal error(比如编译错误)。下面是一个使用自定义错误处理函数的列子:

<?php
error_reporting (0);
function error_handler ($error_level, $error_message, $file, $line) {
  $EXIT = FALSE;
  switch ($error_level) {
    case E_NOTICE:
    case E_USER_NOTICE:
      $error_type = 'Notice';
      break;
    case E_WARNING:
    case E_USER_WARNING:
      $error_type = 'Warning';
      break;
    case E_ERROR:
    case E_USER_ERROR:
      $error_type = 'Fatal Error';
      $EXIT = TRUE;
      break;
    default:
      $error_type = 'Unknown';
      $EXIT = TRUE;
      break;
  }
  printf ("%s: %s in %s on line %d\n", $error_type, $error_message, $file, $line);
 
  if ($EXIT) {
    die();
  }
}
set_error_handler ('error_handler');
 
//new NonExist();
echo $novar;
echo 3/0;
trigger_error ('Trigger a fatal error', E_USER_ERROR);
new NonExist();
?>

执行此脚本可以得到下面的输出:

Notice: Undefined variable: novar in /your/php_demo_file.php on line 40

Warning: Division by zero in /your/php_demo_file.php on line 41

Fatal Error: Trigger a fatal error in /your/php_demo_file.php on line 42

可以看到,最后的“new NoExistClass()”的异常,没有被自定义的错误处理函数捕获。

最后,捎带提一下,set_exception_handler注册顶层的异常处理,在web一用中,可以设定一下,然后统一的跳转到错误处理页面。

PHP 相关文章推荐
PHP远程连接MYSQL数据库非常慢的解决方法
Jul 05 PHP
PHP创建桌面快捷方式的实例代码
Feb 17 PHP
PHP中使用Session配合Javascript实现文件上传进度条功能
Oct 15 PHP
PHP扩展模块memcached长连接使用方法分析
Dec 24 PHP
php多次include后导致全局变量global失效的解决方法
Feb 28 PHP
PHP实现的线索二叉树及二叉树遍历方法详解
Apr 25 PHP
Linux系统中为php添加pcntl扩展
Aug 28 PHP
总结一些PHP中好用但又容易忽略的小知识
Jun 02 PHP
PHP读取word文档的方法分析【基于COM组件】
Aug 01 PHP
laravel中短信发送验证码的实现方法
Apr 25 PHP
PHP实现的多维数组去重操作示例
Jul 21 PHP
深入学习微信网址链接解封的防封原理visit_type
Aug 15 PHP
PHP预定义变量9大超全局数组用法详解
Apr 23 #PHP
php构造函数与析构函数
Apr 23 #PHP
浅谈PHP中的
Apr 23 #PHP
简单谈谈PHP中的include、include_once、require以及require_once语句
Apr 23 #PHP
浅析Yii2 gridview实现批量删除教程
Apr 22 #PHP
浅析Yii2 GridView 日期格式化并实现日期可搜索教程
Apr 22 #PHP
浅析Yii2 GridView实现下拉搜索教程
Apr 22 #PHP
You might like
一些星际专用术语解释
2020/03/04 星际争霸
thinkphp3.2.2前后台公用类架构问题分析
2014/11/25 PHP
laravel 5.3中自定义加密服务的方案详解
2017/05/09 PHP
php实现保存周期为1天的购物车类
2017/07/07 PHP
PHP+redis实现的悲观锁机制示例
2018/06/12 PHP
Thinkphp 框架扩展之行为扩展原理与实现方法分析
2020/04/23 PHP
为Javascript中的String对象添加去除左右空格的方法(示例代码)
2013/11/30 Javascript
node.js中的querystring.stringify方法使用说明
2014/12/10 Javascript
AngularJS中的过滤器使用详解
2015/06/16 Javascript
jQuery实现复选框批量选择与反选的方法
2015/06/17 Javascript
AngularJS学习笔记之依赖注入详解
2016/05/16 Javascript
网页挂马方式整理及详细介绍
2016/11/03 Javascript
JavaScript中捕获/阻止捕获、冒泡/阻止冒泡方法
2016/12/07 Javascript
jQuery实现页面倒计时并刷新效果
2017/03/13 Javascript
jQuery树插件zTree使用方法详解
2017/05/02 jQuery
Vue一个案例引发的递归组件的使用详解
2018/11/15 Javascript
JavaScript类型相关的常用操作总结
2019/02/14 Javascript
使用Python导出Excel图表以及导出为图片的方法
2015/11/07 Python
基于Python中单例模式的几种实现方式及优化详解
2018/01/09 Python
python: 自动安装缺失库文件的方法
2018/10/22 Python
python批量爬取下载抖音视频
2019/06/17 Python
python3.6根据m3u8下载mp4视频
2019/06/17 Python
python绘制已知点的坐标的直线实例
2019/07/04 Python
pip安装python库的方法总结
2019/08/02 Python
利用rest framework搭建Django API过程解析
2019/08/31 Python
Expedia爱尔兰:酒店、机票、租车及廉价假期
2017/01/02 全球购物
社区工作感言
2014/02/21 职场文书
甜品店创业计划书
2014/08/14 职场文书
放飞梦想演讲稿600字
2014/08/26 职场文书
2014年保洁工作总结
2014/11/24 职场文书
2015年女职工工作总结
2015/05/15 职场文书
2015年医院后勤工作总结
2015/05/20 职场文书
检讨书怎么写?
2019/06/21 职场文书
手把手教你用SpringBoot将文件打包成zip存放或导出
2021/06/11 Java/Android
分布式Redis Cluster集群搭建与Redis基本用法
2022/02/24 Redis
P站美图推荐——变身女主角特辑
2022/03/20 日漫