除非你打算做一些有意义的事情,否则你不应该抓住异常。
“有意义的东西”可能是其中之一:
处理异常
最明显的有意义的操作是处理异常,例如,通过显示错误消息并中止操作:
try {
$connect = new CONNECT($db, $user, $password, $driver, $host);
}
catch (Exception $e) {
echo "Error while connecting to database!";
die;
}
日志记录或部分清理
有时您不知道如何在特定上下文中正确处理异常;也许您缺乏有关“大局”的信息,但是您确实希望尽可能接近故障发生的位置。在这种情况下,您可能需要捕获、记录和重新抛出:
try {
$connect = new CONNECT($db, $user, $password, $driver, $host);
}
catch (Exception $e) {
logException($e); // does something
throw $e;
}
一个相关的场景是,您处于正确的位置,可以对失败的操作执行一些清理,但不能决定如何在顶层处理故障。在早期的 PHP 版本中,这将实现为
$connect = new CONNECT($db, $user, $password, $driver, $host);
try {
$connect->insertSomeRecord();
}
catch (Exception $e) {
$connect->disconnect(); // we don't want to keep the connection open anymore
throw $e; // but we also don't know how to respond to the failure
}
PHP 5.5 引入了关键字,因此对于清理方案,现在有另一种方法可以解决这个问题。如果清理代码无论发生什么(即错误和成功)都需要运行,现在可以这样做,同时透明地允许任何抛出的异常传播:finally
$connect = new CONNECT($db, $user, $password, $driver, $host);
try {
$connect->insertSomeRecord();
}
finally {
$connect->disconnect(); // no matter what
}
错误抽象(带异常链接)
第三种情况是,您希望将许多可能的故障逻辑地分组到一个更大的保护伞下。逻辑分组示例:
class ComponentInitException extends Exception {
// public constructors etc as in Exception
}
class Component {
public function __construct() {
try {
$connect = new CONNECT($db, $user, $password, $driver, $host);
}
catch (Exception $e) {
throw new ComponentInitException($e->getMessage(), $e->getCode(), $e);
}
}
}
在这种情况下,您不希望 的用户知道它是使用数据库连接实现的(也许您希望保持选项打开并在将来使用基于文件的存储)。因此,您的规范会说“在初始化失败的情况下,将抛出”。这允许 的使用者捕获预期类型的异常,同时还允许调试代码访问所有(依赖于实现的)详细信息。Component
Component
ComponentInitException
Component
提供更丰富的上下文(带异常链接)
最后,在某些情况下,您可能希望为异常提供更多上下文。在这种情况下,将异常包装在另一个异常中是有意义的,该异常包含有关发生错误时您尝试执行的操作的更多信息。例如:
class FileOperation {
public static function copyFiles() {
try {
$copier = new FileCopier(); // the constructor may throw
// this may throw if the files do no not exist
$copier->ensureSourceFilesExist();
// this may throw if the directory cannot be created
$copier->createTargetDirectory();
// this may throw if copying a file fails
$copier->performCopy();
}
catch (Exception $e) {
throw new Exception("Could not perform copy operation.", 0, $e);
}
}
}
这种情况与上述情况类似(该示例可能不是最好的示例),但它说明了提供更多上下文的要点:如果引发异常,它会告诉我们文件复制失败。但是为什么它失败了呢?此信息在包装的异常中提供(如果示例要复杂得多,则其中可能有多个级别)。
如果您考虑一个场景,例如,创建一个对象会导致文件被复制,因为用户配置文件存储在文件中并且它支持事务语义,则可以“撤消”更改,因为它们仅在配置文件的副本上执行,直到您提交为止。UserProfile
在这种情况下,如果您这样做了
try {
$profile = UserProfile::getInstance();
}
结果捕获了“无法创建目标目录”异常错误,您将有权感到困惑。将此“核心”异常包装在提供上下文的其他异常层中将使错误更容易处理(“创建配置文件复制失败” - > “文件复制操作失败” - >“无法创建目标目录”)。