错误和异常
一直对错误和异常的区别很模糊,究竟什么时候该用错误,什么时候该用异常。我还是问了我的一个朋友,下面引用一下他精炼的回答:
error表示是一种严重问题,但致命的错误很难去处理,比如内存溢出等。
exception表示一种可以解决的问题,是人为导致的问题,比如数组下标越界,读取文件不存在等。
PHP错误
PHP错误有三种方式处理,分别是die语句,自定义错误和错误触发器,错误报告。
-
die语句
一个简单的例子,若打开文件不存在,用die语句打印信息,并终止脚本运行。
<?php
if(!file_exists("hello.txt"))
{
die("文件不存在");
}
else
{
$file=fopen("hello.txt", "r");
}
?>
-
自定义错误和错误触发器
有时候终止脚本运行不是一个好的选择,我们可以自定义错误函数。错误函数原型为
error_function(error_level,error_message,error_file,error_line,error_context)
错误函数参数说明:
参数 | 说明 |
---|---|
error_level | 必需。错误级别,级别说明参见下面表格 |
error_message | 必须。错误信息 |
error_file | 可选,错误文件 |
error_line | 可选,错误行 |
error_context | 可选,一个数组,包含了错误发生时的变量和他们的值 |
错误级别说明:
值 | 常量 | 说明 |
---|---|---|
2 | E_WARNING | 非致命的runtime错误,不暂停脚本执行 |
8 | E_NOTICE | runtime通知,脚本发现可能有错误时发生,也可能在运行时发生 |
256 | E_USER_ERROR | 致命的用户生成的错误 |
512 | E_USER_WARNING | 非致命的用户生成的警告 |
1024 | E_USER_NOTICE | 用户生成的通知 |
4096 | E_RECOVERABLE_ERROR | 可捕获的致命错误 |
8191 | E_ALL | 所有错误和警告 |
我们写好了错误函数,可以把它定义为错误处理程序,通过set_error_handler(function, [error_level])实现,第一个参数是写好的错误函数,第二个参数是可选的错误级别。
设置好了错误处理程序,还需要一个触发器来触发它,通过trigger_error($err_msg, [error_level])实现,第一个参数是错误信息,第二个是可选的错误级别(仅仅包括用户级别,值为256或512或1024)。
一个简单的自定义错误处理和触发器的程序为:
<?php
// 错误处理函数
function customError($errno, $errstr)
{
echo "<b>Error:</b> [$errno] $errstr<br>";
}
//设置错误处理函数
set_error_handler("customError", E_USER_WARNING);
$test=2;
if($test>1)
{
//触发错误处理程序
trigger_error("变量值必须小于等于1", E_USER_WARNING);
}
?>
错误报告
默认情况下,脚本根据php.ini文件的error_log配置,PHP向服务器的记录系统或文件发送错误记录。通过error_log函数可以向指定的目的地发送记录。比如你通过向电子邮箱发送错误记录来通知用户:
<?php
function customError($errno, $errstr)
{
echo "<b>Error:</b> [$errno] $errstr<br>";
error_log("Error: [$errno] $errstr",1,"someone@example.com","From: webmaster@example.com");
}
set_error_handler("customError", E_USER_WARNING);
$test=2;
if($test>1)
{
trigger_error("变量值必须小于等于1", E_USER_WARNING);
}
?>
PHP异常
PHP异常可以用基本的异常处理,也可以自定义异常类处理以及多个异常的处理。
-
基本的异常处理
和c++等面向对象语言的异常处理相似,使用try-throw-catch代码结构。
<?php
// 创建一个有异常处理的函数
function checkNum($number)
{
if($number>1)
{
//抛出异常
throw new Exception("变量值必须小于等于 1");
}
return true;
}
// 在 try 块 触发异常
try
{
checkNum(2);
// 如果抛出异常,以下文本不会输出
echo '如果输出该内容,说明 $number 变量小于等于1';
}
// 捕获异常
catch(Exception $e)
{
echo 'Message: ' .$e->getMessage();
}
?>
抛出的异常对象有很多方法,比如getMessage, getLine, getFile, getCode等,具体可以查看PHP官方中文手册。
-
自定义异常类
创建的异常类必须继承基类Exception,这样它可以继承Exception所有的上述的方法。
<?php
class customException extends Exception
{
public function errorMessage()
{
// 错误信息
$errorMsg = '错误行号 '.$this->getLine().' in '.$this->getFile()
.': <b>'.$this->getMessage().'</b> 不是一个合法的 E-Mail 地址';
return $errorMsg;
}
}
$email = "someone@example...com";
try
{
// 检测邮箱
if(filter_var($email, FILTER_VALIDATE_EMAIL) === FALSE)
{
// 如果是个不合法的邮箱地址,抛出异常
throw new customException($email);
}
}
catch (customException $e)
{
//display custom message
echo $e->errorMessage();
}
?>
以上代码输出错误信息。
-
多个异常
每个异常的抛出需要一个对应的catch块来处理,在一个异常里也可以抛出另外一个异常。你可以设置一个顶层异常处理器来处理所有未被捕获的异常。和设置错误处理函数类似,设置顶层异常处理器 set_exception_handler(exception_function)。
<?php
function myException($exception)
{
echo "<b>Exception:</b> " , $exception->getMessage();
}
set_exception_handler('myException');
throw new Exception('Uncaught Exception occurred');
?>
如上所示,抛出的未捕获的异常将由顶层异常处理函数myException来处理。
最后总结一下异常的规则:
1.可能触发异常的代码放在try内。
2.每一个throw抛出异常都要对应一个catch捕获异常,
3.可以在catch块的代码里再次抛出异常
4.可以设置一个顶层异常处理函数来处理未被捕获的异常。