推荐:《PHP7教程》
PHP语言简单的原因之一就是PHP的错误处理机制,随着PHP语言越来越现代化,也出现了异常,这篇博文就是简单说下错误和异常,以便系统的理解,另外对于任何一种语言来说,异常的存在是具备共性的,所以学习一门语言理解异常机制是必不可少的.
什么是错误
当PHP语言遇到异常的情况(比如数据库连接不上或者函数参数传递错误),则会报出一些错误,错误可以分为多种类型,除了E_ERROR和E_CORE_ERROR错误,其它错误不会终止程序运行.
PHP让人觉得简单的原因就在于程序不会频繁的报错,给人一种编写流畅和方便的错觉.
也正因为这个原因PHP程序的严谨性和准确性差了不少,比如mysql_fetch_array查询遇到网络错误返回FALSE的时候(程序没有终止运行),假如调用程序认为查询没有匹配的数据,则这个程序本质是错误的.
通过 php.ini的指令 error_reporting或者动态调用 error_reporting()函数我们可以选择报告什么类型的错误,通过 display_errors指令则可以控制错误是否在线输出.而 error_log指令可以控制将错误输出到日志中.
如何正确使用错误
不管是系统函数或者是自定义函数,假如内部遇到错误,如何告之调用者呢?一般是通过函数返回 TRUE或者 FALSE来表明.这种处理方式有几个弊端:
● 调用者只知道发生了错误,但是返回的错误信息太少,且缺乏错误类型的说明
● 程序处理逻辑和错误处理混杂在一块,产生的代码会非常的不清晰.
一个小技巧: error_get_last()函数会返回最近错误产生的具体原因.
最佳实践:
● set_error_handler()函数来托管所有的错误
● trigger_error()函数可以触发自定义错误,可以用来在函数中代替 return 语句
● 将所有的错误输出到日志中,同时定义错误类型
● 对用户显示错误,比如将错误以一种更友好的方式返回给用户
● 生产环境下 display_errors指令要关闭,开发环境则该指令打开
老牌的PHP框架 Codeigniter处理错误的方式可以借鉴
`function _error_handler($severity, $message, $filepath, $line) { $is_error = (((E_ERROR | E_COMPILE_ERROR | E_CORE_ERROR | E_USER_ERROR) & $severity) === $severity); //输出500错误HTTP状态码 if ($is_error) { set_status_header(500); } //对于不需要处理的错误则直接中断 if (($severity & error_reporting()) !== $severity) { return; } //将所有的错误记录到日志中 $_error =& load_class('Exceptions', 'core'); $_error->log_exception($severity, $message, $filepath, $line); //友好的输出所有错误 if (str_ireplace(array('off', 'none', 'no', 'false', 'null'), '', ini_get('display_errors'))){ $_error->show_php_error($severity, $message, $filepath, $line); } //假如致命错误则直接退出 if ($is_error) { exit(1); } } set_error_handler('_error_handler');`
什么是异常
异常也是一个错误,它具备以下的特点:
● 异常可以自定义,SPL提供了很多类型的异常,你也可以扩展它
● 异常最常规的动作就是捕获,这样开发者就能根据具体的错误进行后续处理.比如可以根据异常的上下文给用户返回友好的提示.或者继续抛出一个异常,让上游的程序去处理.假如还是没有捕获异常,那么程序就直接终止了.
● 异常另外个动作就是抛出,假如通过函数编写业务逻辑,遇到意外的情况,可以直接扔出一个异常.
● 异常可以被代码一层一层捕获,假如最外层的程序还没有捕获,则代码直接终止运行
● PHP中的异常假如不能捕获,则作为致命错误写入到系统错误日志中
通过直观的代码来说明下:
`function inverse($x) { if ($x < 10) { throw new Exception('x<10'); } elseif ($x >= 10 and $x < 100) { throw new LogicException('x>=10 and x<100'); } return $x; } try { echo inverse(2)."n"; } catch (LogicException $e) { echo 'Caught LogicException: ', $e->getMessage(), "n"; } catch (Exception $e) { echo 'Caught Exception: ', $e->getMessage(), "n"; throw $e; }`
异常的最佳实践
● 异常可以让代码更加清晰,让开发者专注于业务逻辑的编写.
● 构建可扩展的异常是非常有技术性的,难道SPL异常还做的不够吗?
● 捕获异常应该仅仅捕获本层能处理的异常,对于不能处理的异常则让上游的代码处理.
PHP7中的异常
PHP7鼓励使用异常来代替错误,但是不可能一下子推翻错误处理机制,需要兼容,所以只能慢慢过渡.
但是可以通过变通的方式来统一使用异常
● Error异常
PHP中定义了一个 Error异常,注意这个异常和 Exception是并列的,
当打开严格模式的时候,PHP7中很多的错误是被 Error异常抛出的.这样就能统一使用异常了.
`declare (strict_types = 1); function add(int $a, int $b) { return $a + $b; } try { echo add("3", "4"); } catch (TypeError $e) { //TypeError继承自Error echo $e->getMessage(); }`
● ErrorException
ErrorException继承自 Exception.
我们可以通过 set_error_handler()函数将所有的错误转换成 ErrorException.这样就能愉快的统一使用异常了.