更新:
从 PHP 7.1 开始,此功能可用。
语法是:
try
{
// Some code...
}
catch(AError | BError $e)
{
// Handle exceptions
}
catch(Exception $e)
{
// Handle the general case
}
文档:https://www.php.net/manual/en/language.exceptions.php#example-294
RFC:https://wiki.php.net/rfc/multiple-catch
提交:https://github.com/php/php-src/commit/0aed2cc2a440e7be17552cc669d71fdd24d1204a
对于 7.1 之前的 PHP:
尽管这些其他答案说了什么,您可以在同一块中捕获 AError 和 BError(如果您是定义异常的人,这会更容易一些)。即使存在您想要“通过”的例外情况,您仍然应该能够定义一个层次结构来满足您的需求。
abstract class MyExceptions extends Exception {}
abstract class LetterError extends MyExceptions {}
class AError extends LetterError {}
class BError extends LetterError {}
然后:
catch(LetterError $e){
//voodoo
}
如您所见here 和here,即使是SPL 默认异常也有一个可以利用的层次结构。另外,如PHP Manual中所述:
当抛出异常时,语句后面的代码不会
执行,PHP 将尝试找到第一个匹配的 catch 块。
这意味着你也可以拥有
class CError extends LetterError {}
您需要以不同于AError 或BError 的方式处理它,因此您的catch 语句如下所示:
catch(CError $e){
//voodoo
}
catch(LetterError $e){
//voodoo
}
如果您遇到的情况是,有 20 个或更多异常合法地属于同一个超类,并且您需要以一种方式处理其中的 5 个(或任何大型组),而其余的则以另一种方式处理,您仍然可以这样做。
interface Group1 {}
class AError extends LetterError implements Group1 {}
class BError extends LetterError implements Group1 {}
然后:
catch (Group1 $e) {}
在处理异常时使用 OOP 非常强大。使用 get_class 或 instanceof 之类的东西是黑客行为,应尽可能避免使用。
我想添加的另一个解决方案是将异常处理功能放在它自己的方法中。
你可以有
function handleExceptionMethod1(Exception $e)
{
//voodoo
}
function handleExceptionMethod2(Exception $e)
{
//voodoo
}
假设您绝对无法控制异常类层次结构或接口(并且几乎总是将有办法),您可以执行以下操作:
try
{
stuff()
}
catch(ExceptionA $e)
{
$this->handleExceptionMethod1($e);
}
catch(ExceptionB $e)
{
$this->handleExceptionMethod1($e);
}
catch(ExceptionC $e)
{
$this->handleExceptionMethod1($e);
}
catch(Exception $e)
{
$this->handleExceptionMethod2($e);
}
这样,如果您的异常处理机制需要更改,您仍然只有一个必须修改的代码位置,并且您在 OOP 的一般结构中工作。