php7的错误与异常处理

  • 来源:网络
  • 更新日期:2020-08-12

摘要:错误错误是写程序中不可避免的,如何去发现定位错误及改正更是一门学问。今天和大家聊聊php7之后的错误与异常。错误的分类一般错误分为语法错误、运行时错误、逻辑错误。语法

错误

错误是写程序中不可避免的,如何去发现定位错误及改正更是一门学问。今天和大家聊聊php7之后的错误与异常。

错误的分类

一般错误分为语法错误、运行时错误、逻辑错误。语法以及运行时错误很容易去排查,但逻辑错误就不是那么简单的了。鄙人在判断的时候也有多次将比较运算==写成了赋值运算=,像我这种就很难去发现。一般随着经验的丰富(踩坑踩多了),会越来越容易来找到逻辑错误。

控制错误的输出

一般建议开发阶段开启错误输出方便及时发现错误,上线阶段为了安全性及美观建议关闭错误输出。

php提供了一个参数display_errors用于控制错误向浏览器、cli的输出。有两种方式可以来修改它,通过修改php.ini文件或使用ini_set函数来设置。下面展示使用ini_set来设置屏蔽错误输出的代码。

<?php
ini_set('display_errors', 0);

echo $a;
echo 0;
date();
echo 1;

该程序会正常输出01,但不会输出错误信息。

错误报告级别

一般分为4大类

parse语法错误

error致命错误

warning警告级别错误

notice注意级别错误

在php.ini配置文件中,有选项error_reporting,该选项用来控制输出何种级别的错误。常见的有E_ALL、E_WARNING、E_NOTICE。

可以通过函数error_reporting()来动态控制错误级别的输出。

# 输出所有级别错误
error_reporting(E_ALL);
# 输出所有级别除了E_NOTICE
error_reporting(E_ALL & ~E_NOTICE);
# 输出所有级别除了E_NOTICE和E_WARNING
error_reporting(E_ALL & ~(E_NOTICE | E_WARNING));

display_errors与error_reporting的区别

display_errors是用来控制错误是否输出,而error_reporting是用来控制输出何种级别的错误。通常他们会搭配使用

ini_set('display_errors', 0); // 关闭错误输出
error_reporting(E_ALL);  // 输出所有级别的错误信息

上面的配置一般是上线阶段,屏蔽所有的错误输出,但是记录错误到php的错误日志里。错误日志的路径是由选项error_log 决定的。

如果设置error_reporting为0值,则不输出错误也不记录错误日志。

错误对于程序的影响

当程序有语法错误时,程序是不会执行的。当有error级别的错误时,程序就会停止向下执行。notice、及warning级别错误时不会影响程序的向下执行的。

error_reporting(E_ALL);  // 输出所有级别的错误信息

echo $a;
echo 0;
date();
echo 1;
new a;
echo 2;

该程序会输出0和1,但不会输出2。

用户自定义错误

trigger_error()函数可以生产一个用户级别的错误。错误级别有E_USER_ERROR 、E_USER_WARNING、E_USER_NOTICE 等

<?php

trigger_error('这是notice级别错误', E_USER_NOTICE);
trigger_error('warning错误', E_USER_WARNING);
trigger_error('deprecated', E_USER_DEPRECATED);
trigger_error('error错误', E_USER_ERROR);

自定义错误处理

set_error_handler — 设置用户自定义的错误处理函数,该函数原型如下:

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

该函数首个参数是一个回调函数,原型如下:

handler ( int $errno , string $errstr [, string $errfile [, int $errline [, array $errcontext ]]] ) : bool

error 错误级别

errstr 错误信息

errfile 发送错误的文件

errline 错误出现的行号

以下级别的错误不能由用户定义的函数来处理: E_ERROR、 E_PARSE、 E_CORE_ERROR、 E_CORE_WARNING、 E_COMPILE_ERROR、 E_COMPILE_WARNING,和在调用 set_error_handler() 函数所在文件中产生的大多数 E_STRICT。

<?php
 
define('DEBUG', false);
 
set_error_handler('error_handler');
 
if (DEBUG) {
    ini_set('display_errors', 'On');
} else {
    ini_set('display_errors', 'Off');
}
 
function error_handler($errLevel, $errInfo, $errFile, $errLine)
{
    echo "ErrorLevel:$errLevel: $errInfo In $errFile ON $errLine" . PHP_EOL;
}
 
echo $a;

异常

异常和错误时有区别的,错误一般是指我们能控制的问题,比如变量名写错了,或者判断条件写的不错,导致死循环。而异常通常指那些难以控制的、意料外的错误,比如mysql连接不上,文件句柄打开失败等情况。

php的异常也是经典的try catch finally,但和一般的异常处理不一样的是,绝大部分的异常需要自行抛出。抛出异常使用throw关键字完成。php也支持捕获多个异常。

<?php

class MyError extends Exception
{
    public function printErr ()
    {
        echo '出错啦'.PHP_EOL;
    }
}

class YourError extends Exception
{
    public function printErr ()
    {
        echo 'errors'.PHP_EOL;
    }
}

try {
    if (mt_rand(0,1)) {
        throw new MyError('错误');
    } else {
        throw new YourError('错误');
    }


} catch (MyError $e) {
    $e->printErr();
} catch (YourError $e) {
    $e->printErr();
} finally {
    echo '不管有没有异常,我都会被执行'.PHP_EOL;
}

php的异常如果没有捕获,则会报Fatal Error错误,程序不会继续向下执行。

PHP 7 错误处理

PHP 7 改变了大多数错误的报告方式。不同于传统(PHP 5)的错误报告机制,现在大多数错误被作为 Error 异常抛出。 Error 和 Exception 都实现了 Throwable 接口

异常处理

设置默认的异常处理程序,有try/catch捕获的话这个异常函数就不会执行,反之就会执行异常处理函数,而且执行的话,脚本将不会继续执行。

php使用set_exception_handler来设置用户自定义的异常处理函数 ,函数原型如下:

set_exception_handler ( callable $exception_handler ) : callable

回调函数的原型如下:

handler ( Throwable $ex ) : void

下面,我们来写一个异常处理函数

<?php

class MyError extends Exception
{
    public function printErr ()
    {
        echo '出错啦'.PHP_EOL;
    }
}

class YourError extends Exception
{
    public function printErr ()
    {
        echo 'errors'.PHP_EOL;
    }
}

try {
    if (mt_rand(0,1)) {
        throw new MyError('错误');
    } else {
        throw new YourError('错误');
    }


} catch (MyError $e) {
    $e->printErr();
} catch (YourError $e) {
    $e->printErr();
} finally {
    echo '不管有没有异常,我都会被执行'.PHP_EOL;
}

统一处理错误与异常

学完了错误与异常,知道如何去使用自定义错误处理和异常处理,现在我们就可以统一处理错误与异常了。

<?php
 
class Errors
{
    // 处理非致命错误
    static function errorHandle($errLevel, $errInfo, $errFile, $errLine)
    {
        echo '错误:'.PHP_EOL;
        print_r(['file' => $errFile, 'level' => $errLevel, 'line' => $errLine, 'info' => $errInfo]);
    }
 
    // 处理致命错误及异常
    static function exceptionHandle(Throwable $ex)
    {
        echo '异常:'.PHP_EOL;
        print_r(['file' => $ex->getFile(), 'level' => $ex->getCode(), 'line' => $ex->getLine(), 'info' => $ex->getMessage()]);
    }
}
 
set_error_handler(['Errors', 'errorHandle']);
set_exception_handler(['Errors', 'exceptionHandle']);