我应该抛出不同的异常类型吗?

时间:2013-09-16 09:03:57

标签: php exception-handling phpunit

当故意抛出异常时,我总是使用PHP的基本异常类,只是使用不同的消息,例如:

<?php
if($a < $b){
  throw new Exception('a is smaller than b');
}elseif($a > $b){
  throw new Exception('a is larger than b');
}

然而,最近我注意到有些人会抛出不同类别的异常,例如:

<?php
if($a < $b){
  throw new aIsSmallerThanBException();
}elseif($a > $b){
  throw new aIsLargerThanBException();
}

这两种方法有哪些优点/缺点?我应该以第二种方式做吗?我注意到在使用PHPUnit进行单元测试时,您可以使用setExpectedException()告诉它您正在使用某种异常类。这是否意味着第二种方法在使代码可测试方面更好?

谢谢!

3 个答案:

答案 0 :(得分:3)

这些不同的异常可以使流程更容易,因为您可以同时捕获多种类型的异常。看看这个:

function mustBeEqual($a, $b){
  if($a < $b){
    throw new aIsSmallerThanBException()('a is smaller than b');
  }elseif($a > $b){
    throw new aIsLargerThanBException()('a is larger than b');
  } else {
    //Everything is ok, proceed.
  }
}

try {
  mustBeEqual(3, 4);
} catch (isLargerThanBException $e){
  echo "A is greater than B" ;
} catch (isSmallerThanBException $e){
  echo "A is smaller than B" ;
} catch (Exception e){
  echo "Default exception was triggered: " . $e->getMessage() ;
}

首先,您希望捕获一些特定的异常,然后使用默认异常只是为了确保捕获它。每个特定异常都是Exception的派生类,并继承getCodegetMessage等方法。

我还可以举例说明它的工作原理:

  1. 在try-catch块中调用函数。函数抛出异常。
  2. 尝试抓住ConnectionTimeoutException。例外不是这种类型。转到下一个catch
  3. 尝试抓住AccessForbiddenException。例外是这种类型。通知用户有关此问题。
  4. 我们不会移动到下一个捕获,因为异常已被捕获。就是这样。

答案 1 :(得分:2)

这是一个选择性的问题。不同的异常类可以由不同的catch子句处理,因此可以根据导致异常的原因运行特定的代码。当您有多个嵌套级别的try-catch块时,这一点尤其重要。

另一件事是代码可读性和更容易调试。 OutOfBoundsException告诉调试人员,普通Exception没有。 SPL定义了一堆有用的exception classes,涵盖了大多数常见问题。使用这些SPL异常,您可以清楚地了解抛出异常的原因。另一个人只需要查找适当的php.net页面。

答案 2 :(得分:1)

在使代码可测试方面,您可以使用第二个。优点是,一旦存在该异常,您就可以在任何脚本中使用该异常。如果是第一个,那么每次要使用异常时都应该声明新的异常throw new Exception('....');