PHP中的异常处理机制深入讲解


Posted in PHP onNovember 10, 2020

1、异常概述

异常(Exception)是一种错误处理机制,用于在指定的错误发生时改变脚本的正常流程。

当异常被触发时,当前代码状态被保存,代码执行被切换到预定义的异常处理器函数(如果有)

根据情况,处理器也许会从保存的代码状态重新开始执行代码,终止脚本执行,或从代码中另外的位置继续执行脚本

2、异常的基本使用

当异常被抛出时,其后的代码不会继续执行,PHP 会尝试查找匹配的 "catch" 代码块。

如果异常没有被捕获,而且又没用使用 set_exception_handler() 作相应的处理的话,那么将发生一个严重的错误(致命错误),并且输出 "Uncaught Exception" (未捕获异常)的错误消息。
尝试抛出一个异常,同时不去捕获它

//create function with an exception
function checkNum($number){
 if($number>1){
  throw new Exception("Value must be 1 or below");
 }
}
//trigger exception
checkNum(2);

上面的代码会获得类似这样的一个错误:

Fatal error: Uncaught exception 'Exception' with message 'Value must be 1 or below' in C:\webfolder\test.php:6
Stack trace: #0 C:\webfolder\test.php(12):checkNum(28) #1 {main} thrown in C:\webfolder\test.php on line 6

注意:PHP默认是警告模式,如果需要对系统错误使用异常处理机制,则要在处理代码之前设置错误处理模式

set_error_handler(function(){
 throw new Exception('错误!');
});

3、Try, throw 和 catch

正确的处理程序应当包括:

Try - 使用异常的函数应该位于 "try" 代码块内。如果没有触发异常,则代码将照常继续执行。但是如果异常被触发,会抛出一个异常。

Throw - 这里规定如何触发异常。每一个 "throw" 必须对应至少一个 "catch"

Catch - "catch" 代码块会捕获异常,并创建一个包含异常信息的对象

//创建可抛出一个异常的函数
function checkNum($number){
 if($number>1){
  throw new Exception("Value must be 1 or below");
 }
}
//在 "try" 代码块中触发异常
try{
 checkNum(2);
}catch(Exception $e) { //捕获异常
 echo 'File: '.$e->getFile().' line: '.$e->getLine().'<br>';
 die('Message: '.$e->getMessage());
}

上面代码将获得类似这样一个错误:

File: E:\webdev\www\pdo\3.php line: 7
Message: Value must be 1 or below

代码解析:创建 checkNum() 函数。它检测数字是否大于 1。如果是,则抛出一个异常。在 "try" 代码块中调用 checkNum(),函数checkNum() 函数中的异常被抛出,"catch" 代码块接收到该异常,并创建一个包含异常信息的对象 ($e)。通过从这个 exception 对象输出来自该异常的错误消息

不过,为了遵循“每个 throw 必须对应一个 catch”的原则,可以设置一个顶层的异常处理器来处理漏掉的错误,见第7点

4、创建自定义的异常处理器

创建一个专门的类,当 PHP 中发生异常时,可调用其函数。该类必须是 exception 类的一个扩展。这个自定义的exception 类继承了 PHP 的 exception 类的所有属性,可向其添加自定义的函数。

class customException extends Exception{
 public function errorMessage(){
  return 'Error on line '.$this->getLine().' in '.$this->getFile().': <b>'.$this->getMessage().'</b> is not a valid E-Mail address';
 }
}
$email = "someone@example...com";
try{
 //使用PHP过滤器验证邮箱有效性
 if(filter_var($email, FILTER_VALIDATE_EMAIL) === FALSE){ 
  throw new customException($email);
 }
}catch (customException $e){
 echo $e->errorMessage();
}

这个新的类是旧的 exception 类的副本,外加 errorMessage() 函数。正因为它是旧类的副本,因此它从旧类继承了属性和方法,我们可以使用 exception 类的方法,比如 getLine() 、 getFile() 以及 getMessage()。

上面的代码抛出了一个异常,并通过一个自定义的 exception 类来捕获它:创建 errorMessage() 函数。如果 e-mail 地址不合法,则该函数返回一条错误消息

5、多个异常的捕获

可以为一段脚本使用多个异常,来检测多种情况。

可以使用多个 if..else 代码块,或一个switch 代码块,或者嵌套多个异常。这些异常能够使用不同的 exception 类,并返回不同的错误消息:

class customException extends Exception{
  public function errorMessage(){
    return = 'Error on line '.$this->getLine().' in '.$this->getFile().': <b>'.$this->getMessage().'</b> is not a valid E-Mail address';
  }
}
$email = "someone@example.com";
try{
  if(filter_var($email, FILTER_VALIDATE_EMAIL) === FALSE){
    throw new customException($email);
  }
  //check for "example" in mail address
  if(strpos($email, "example") !== FALSE){
    throw new Exception("$email is an example e-mail");
  }
} catch (customException $e){
  echo $e->errorMessage();
}catch(Exception $e){
  echo $e->getMessage();
}

上面的代码测试了两种条件,如何任何条件不成立,则抛出一个异常:

执行 "try" 代码块,在第一个条件下,不会抛出异常。由于 e-mail 含有字符串 "example",第二个条件会触发异常。"catch" 代码块会捕获异常,并显示恰当的错误消息,如果没有捕获 customException,只捕获了 base exception,则在那里处理异常。

6、重新抛出异常

有时,当异常被抛出时,也许希望以不同于标准的方式对它进行处理。可以在一个 “catch” 代码块中再次抛出异常。脚本应该对用户隐藏系统错误。对程序员来说,系统错误也许很重要,但是用户对它们并不感兴趣。为了让用户更容易使用,可以再次抛出带有对用户比较友好的消息的异常:

class customException extends Exception{
  public function errorMessage(){
    return $this->getMessage().' is not a valid E-Mail address.';
  }
}
$email = "someone@example.com";
try{
  try{
    if(strpos($email, "example") !== FALSE){
      throw new Exception($email);
    }
  }catch(Exception $e){
    //re-throw exception
    throw new customException($email);
  }
}catch (customException $e){
  //display custom message
  echo $e->errorMessage();
}

上面的代码检测在邮件地址中是否含有字符串 "example"。如果有,则再次抛出异常:

把 $email 变量设置为一个有效的邮件地址,但含有字符串 "example"。"try" 代码块包含另一个 "try" 代码块,这样就可以再次抛出异常。由于 e-mail 包含字符串 "example",因此触发异常。"catch" 捕获到该异常,并重新抛出 "customException"。捕获到 "customException",并显示一条错误消息。

如果在其目前的 "try" 代码块中异常没有被捕获,则它将在更高层级上查找 catch 代码块。

7、设置顶层异常处理器

set_exception_handler() 函数可设置处理所有未捕获异常的用户定义函数

function myException($exception){
  echo "<b>Exception:</b> " , $exception->getMessage();
}
set_exception_handler('myException');
throw new Exception('Uncaught Exception occurred');

以上代码的输出应该类似这样:

Exception: Uncaught Exception occurred

在上面的代码中,不存在 "catch" 代码块,而是触发顶层的异常处理程序。应该使用此函数来捕获所有未被捕获的异常。
但是对于系统自动抛出的错误,会先经过set_error_handler处理并抛出异常才能被set_exception_handler处理

function myException($exception){
  echo "<b>Exception:</b> " , $exception->getMessage();
}
set_exception_handler('myException');
set_error_handler(function(){
  throw new Exception('错误!');
});
echo 10/0; //触发被除数不能为0的警告

代码执行结果:Exception: 错误!

总结

到此这篇关于PHP中异常处理机制的文章就介绍到这了,更多相关PHP异常处理机制内容请搜索三水点靠木以前的文章或继续浏览下面的相关文章希望大家以后多多支持三水点靠木!

PHP 相关文章推荐
PHP产生随机字符串函数
Dec 06 PHP
php下使用curl模拟用户登陆的代码
Sep 10 PHP
PHP判断文件是否存在、是否可读、目录是否存在的代码
Oct 03 PHP
ThinkPHP利用PHPMailer实现邮件发送实现代码
Sep 26 PHP
PHP递归复制、移动目录的自定义函数分享
Nov 18 PHP
Codeigniter的一些优秀特性总结
Jan 21 PHP
PHP批量获取网页中所有固定种子链接的方法
Nov 18 PHP
Yii2下session跨域名共存的解决方案
Feb 04 PHP
ubutu 16.04环境下,PHP与mysql数据库,网页登录验证实例讲解
Jul 20 PHP
Eclipse PHPEclipse 配置的具体步骤
Aug 08 PHP
JS操作XML中DTD介绍及使用方法分析
Jul 04 PHP
PHP实现创建以太坊钱包转账等功能
Apr 21 PHP
基于php解决json_encode中文UNICODE转码问题
Nov 10 #PHP
ThinkPHP5分页paginate代码实例解析
Nov 10 #PHP
关于PHP求解三数之和问题详析
Nov 09 #PHP
PHP中isset、empty的用法与区别示例详解
Nov 05 #PHP
PHP后门隐藏的一些技巧总结
Nov 04 #PHP
phpstudy2020搭建站点的实现示例
Oct 30 #PHP
解决PHPstudy Apache无法启动的问题【亲测有效】
Oct 30 #PHP
You might like
用PHP制作静态网站的模板框架(四)
2006/10/09 PHP
php更改目录及子目录下所有的文件后缀扩展名的代码
2010/10/12 PHP
支持中文的php加密解密类代码
2011/11/27 PHP
解析php file_exists无效的解决办法
2013/06/26 PHP
JQuery中判断一个元素下面是否有内容或者有某个标签的判断代码
2012/02/02 Javascript
js给onclick赋值传参数的两种方法
2013/11/25 Javascript
你未必知道的JavaScript和CSS交互的5种方法
2014/04/02 Javascript
轻松创建nodejs服务器(7):阻塞操作的实现
2014/12/18 NodeJs
深入解析AngularJS框架中$scope的作用与生命周期
2016/03/05 Javascript
ES6所改良的javascript“缺陷”问题
2016/08/23 Javascript
jq给页面添加覆盖层遮罩的实例
2017/02/16 Javascript
Javascript中Promise的四种常用方法总结
2017/07/14 Javascript
Angular利用内容投射向组件输入ngForOf模板的方法
2018/03/05 Javascript
Vue多系统切换实现方案
2018/06/05 Javascript
echarts实现地图定时切换散点与多图表级联联动详解
2018/08/07 Javascript
vue 点击按钮增加一行的方法
2018/09/07 Javascript
BootStrap模态框闪退问题实例代码详解
2018/12/10 Javascript
javascript的惯性运动实现代码实例
2019/09/07 Javascript
vue2.* element tabs tab-pane 动态加载组件操作
2020/07/19 Javascript
python SSH模块登录,远程机执行shell命令实例解析
2018/01/12 Python
python之生产者消费者模型实现详解
2019/07/27 Python
简单了解python中的与或非运算
2019/09/18 Python
python队列原理及实现方法示例
2019/11/27 Python
基于python实现ROC曲线绘制广场解析
2020/06/28 Python
HTML5 拖拽批量上传文件的示例代码
2018/03/28 HTML / CSS
前端实现打印图像功能
2019/08/27 HTML / CSS
amazeui页面分析之登录页面的示例代码
2020/08/25 HTML / CSS
CSMA/CD介质访问控制协议
2015/11/17 面试题
Does C# support multiple inheritance? (C#支持多重继承吗)
2012/01/04 面试题
公司前台接待岗位职责
2013/12/03 职场文书
护理个人求职信范文
2014/01/08 职场文书
观看信仰心得体会
2014/09/04 职场文书
交通安全横幅标语
2014/10/07 职场文书
班级班风口号大全
2015/12/25 职场文书
Nginx设置日志打印post请求参数的方法
2021/03/31 Servers
postgresql如何找到表中重复数据的行并删除
2023/05/08 MySQL