在PHP中,用户可以用自定义的异常处理类来扩展 PHP 内置的异常处理类。通过使用扩展异常处理类,开发者可以更好地控制程序的执行流程,提高代码的可维护性和可读性。本文将介绍如何在PHP中使用扩展异常处理类,以及在内置的异常处理类中,哪些属性和方法在子类中是可访问和可继承的。
一、内置的异常处理类
<?php class Exception implements Throwable { protected $message = 'Unknown exception'; // 异常信息 private $string; // __toString 的缓存 protected $code = 0; // 用户自定义异常错误码 protected $file; // 发生异常的源文件名 protected $line; // 发生异常的源代码行号 private $trace; // backtrace private $previous; // 如果是嵌套异常,则是之前的 exception public function __construct($message = '', $code = 0, Throwable $previous = null); final private function __clone(); // 禁止克隆异常。 final public function getMessage(); // 异常信息 final public function getCode(); // 异常错误码 final public function getFile(); // 发生异常的源文件名 final public function getLine(); // 发生异常的源代码行号 final public function getTrace(); // backtrace() 数组 final public function getPrevious(); // 之前的 exception final public function getTraceAsString(); // 已格成化成字符串的 getTrace() 信息 // Overrideable public function __toString(); // 可输出的格式化后的字符串 } ?>
如果使用自定义的类来扩展内置异常处理类,并且要重新定义构造函数的话,建议同时调用 parent::__construct() 来确保所有的变量已赋值。当对象要输出字符串的时候,可以重载 __toString() 并自定义输出的样式。
注意:不能复制 Exception 对象。尝试对 clone Exception 会导致 fatal E_ERROR 错误。
二、扩展内置的异常处理类
<?php /** * 自定义一个异常处理类 */ class MyException extends Exception { // 重定义构造器使 message 变为必须被指定的属性 public function __construct($message, $code = 0, Throwable $previous = null) { // 这里写用户的代码 // 确保所有变量都被正确赋值 parent::__construct($message, $code, $previous); } // 自定义字符串输出的格式 public function __toString() { return __CLASS__ . ": [{$this->code}]: {$this->message}\n"; } public function customFunction() { echo "A custom function for this type of exception\n"; } } /** * 创建一个类,测试该 exception 类 */ class TestException { public $var; const THROW_NONE = 0; const THROW_CUSTOM = 1; const THROW_DEFAULT = 2; function __construct($avalue = self::THROW_NONE) { switch ($avalue) { case self::THROW_CUSTOM: // 抛出自定义异常 throw new MyException('1 is an invalid parameter', 5); break; case self::THROW_DEFAULT: // 抛出默认的异常 throw new Exception('2 is not allowed as a parameter', 6); break; default: // 没有异常的情况下,创建一个对象 $this->var = $avalue; break; } } } // 例子 1 try { $o = new TestException(TestException::THROW_CUSTOM); } catch (MyException $e) { // 捕获异常 echo "Caught my exception\n", $e; $e->customFunction(); } catch (Exception $e) { // 被忽略 echo "Caught Default Exception\n", $e; } // 继续执行后续代码 var_dump($o); // Null echo "\n\n"; // 例子 2 try { $o = new TestException(TestException::THROW_DEFAULT); } catch (MyException $e) { // 不能匹配异常的种类,被忽略 echo "Caught my exception\n", $e; $e->customFunction(); } catch (Exception $e) { // 捕获异常 echo "Caught Default Exception\n", $e; } // 执行后续代码 var_dump($o); // Null echo "\n\n"; // 例子 3 try { $o = new TestException(TestException::THROW_CUSTOM); } catch (Exception $e) { // 捕获异常 echo "Default Exception caught\n", $e; } // 执行后续代码 var_dump($o); // Null echo "\n\n"; // 例子 4 try { $o = new TestException(); } catch (Exception $e) { // 没有异常,被忽略 echo "Default Exception caught\n", $e; } // 执行后续代码 var_dump($o); // TestException echo "\n\n"; ?>