使用 register_shutdown_function() 处理 PHP 中的致命错误

Posted

技术标签:

【中文标题】使用 register_shutdown_function() 处理 PHP 中的致命错误【英文标题】:Handle fatal errors in PHP using register_shutdown_function() 【发布时间】:2011-05-23 13:50:31 【问题描述】:

根据the comment on this answer,可以通过shutdown function 捕获致命错误,而使用set_error_handler() 无法捕获。

但是,我不知道如何确定关闭是由于致命错误还是由于脚本到达结尾而发生的。

此外,调试回溯功能似乎在关闭功能中失效,使得记录发生致命错误的堆栈跟踪变得毫无价值。

所以我的问题是:对致命错误(尤其是未定义的函数调用)做出反应,同时保持创建适当回溯的能力的最佳方法是什么?

【问题讨论】:

【参考方案1】:

这对我有用:

function shutdown() 
    $error = error_get_last();
    if ($error['type'] === E_ERROR) 
        // fatal error has occured
    


register_shutdown_function('shutdown');

spl_autoload_register('foo'); 
// throws a LogicException which is not caught, so triggers a E_ERROR

但是,您可能已经知道了,但只是为了确保:您无法以任何方式从 E_ERROR 中恢复。

至于回溯,你不能...... :( 在大多数致命错误的情况下,尤其是 Undefined function 错误,你并不真正需要它。精确定位文件/行它发生在哪里就足够了。在这种情况下,回溯是无关紧要的。

【讨论】:

您应该在条件前面加上“$error !== null &&”。否则会在以前没有错误时触发。【参考方案2】:

使用 register_shutdown_function 区分致命错误和正确关闭应用程序的一种方法是将一个常量定义为程序的最后一行,然后检查该常量是否已定义:

function fatal_error() 
if ( ! defined(PROGRAM_EXECUTION_SUCCESSFUL)) 
        // fatal error has occurred
    


register_shutdown_function('fatal_error');

define('PROGRAM_EXECUTION_SUCCESSFUL', true);

如果程序结束,它不可能遇到致命错误,所以如果定义了常量,我们知道不运行该函数。

error_get_last() 是一个数组,其中包含有关您应该需要调试的致命错误的所有信息,尽管如前所述,它没有回溯。

一般来说,如果您的 php 程序遇到了致命错误(而不是异常),您希望程序崩溃,以便找到并解决问题。我发现 register_shutdown_function 对于您希望关闭错误报告的生产环境很有用,但需要某种方式在后台记录错误以便您可以响应它。如果出现此类错误,您还可以使用该功能将用户引导至友好的 html 页面,这样您就不会只是提供空白页面。

【讨论】:

【参考方案3】:

获取当前 error_handler 方法只是一个不错的技巧 =)

<?php
register_shutdown_function('__fatalHandler');
function __fatalHandler()

    $error      = error_get_last();

    //check if it's a core/fatal error, otherwise it's a normal shutdown
    if($error !== NULL && $error['type'] === E_ERROR) 
        //Bit hackish, but the set_exception_handler will return the old handler
        function fakeHandler()  
        $handler = set_exception_handler('fakeHandler');
        restore_exception_handler();
        if($handler !== null)  
            call_user_func($handler, new ErrorException($error['message'], $error['type'], 0, $error['file'], $error['line']));
        
        exit;
    

?>

如果你打电话,我也不想注意

<?php
ini_set('display_errors', false);
?>

Php 停止显示错误,否则错误文本将在您的错误处理程序之前发送到客户端

【讨论】:

以上是关于使用 register_shutdown_function() 处理 PHP 中的致命错误的主要内容,如果未能解决你的问题,请参考以下文章

在使用加载数据流步骤的猪中,使用(使用 PigStorage)和不使用它有啥区别?

今目标使用教程 今目标任务使用篇

Qt静态编译时使用OpenSSL有三种方式(不使用,动态使用,静态使用,默认是动态使用)

MySQL db 在按日期排序时使用“使用位置;使用临时;使用文件排序”

使用“使用严格”作为“使用强”的备份

Kettle java脚本组件的使用说明(简单使用升级使用)