可以看到,通过结合使用throw关键字和try-catch语句,我们可以避免错误标记“污染”类方法返回的值。因为“异常”本身就是一种与其它任何对象不同的php内建的类型,不会产生混淆。
如果抛出了一个异常,try语句中的脚本将会停止执行,然后马上转向执行catch语句中的脚本。
如果异常抛出了却没有被捕捉到,就会产生一个fatal error。
处理多个错误
在目前为止异常处理看起来和我们传统的作法—检验返回的错误标识或对象的值没有什么太大区别。让我们将commandmanager处理地更谨慎,并在构造函数中检查command目录是否存在。
index_php5_2.php
<?php
// php 5
require_once('cmd_php5/command.php');
class commandmanager {
private $cmddir = "cmd_php5";
function __construct() {
if (!is_dir($this->cmddir)) {
throw new exception("directory error: $this->cmddir");
}
}
function getcommandobject($cmd) {
$path = "{$this->cmddir}/{$cmd}.php";
if (!file_exists($path)) {
throw new exception("cannot find $path");
}
require_once $path;
if (!class_exists($cmd)) {
throw new exception("class $cmd does not exist");
}
$class = new reflectionclass($cmd);
if (!$class->issubclassof(new reflectionclass('command'))) {
throw new exception("$cmd is not a command");
}
return new $cmd();
}
}
?>
这里有两个地方的调用可能导致程序出错(__construct()和getcommandobject())。尽管如此,我们不需要调整我们的客户代码。你可以在try语句中增添众多内容,然后在catch中统一处理。如果commandmanager 对象的构造函数抛出一个异常,则try语句中的执行中止,然后catch语句被调用捕捉相关的异常。同样地,getcommandobject()也是如此。这样,我们有同时存在两个潜在的引发错误的地方,和一个唯一的语句来处理所有的错误。这让我们的代码看起来更加整洁,又可以满足错误处理的要求。和前面提到的php的传统的错误方法相比,显然很有优势。
index_php5_2.php 后半段
注意:尽管和index_php5.php相比,前半段代码有两个可能出错的地方,这段代码和index_php5.php的后半段完全相同。
<?php
// php 5
try {
$mgr = new commandmanager(); // potential error
$cmd = $mgr->getcommandobject('realcommand');
// another potential error
$cmd->execute();
} catch (exception $e) {
// handle either error here
print $e->getmessage();
exit();
}
?>
还有一个地方我们没有提到。我们怎样区分不同类型的错误?例如,我们可能希望用一种方法来处理找不到目录的错误,而用另一种方法来处理非法的command类。
exception类可以接受一个可选的整型的错误标识,这是在catch语句中区分不同错误类型的一个方法。
index_php5_3.php
<?php
// php 5
require_once('cmd_php5/command.php');
class commandmanager {
private $cmddir = "cmd_php5";
const cmdman_general_error = 1;
const cmdman_illegalclass_error = 2;
function __construct() {
if (!is_dir($this->cmddir)) {
throw new exception("directory error: $this->cmddir", self::cmdman_general_error);
}
}
function getcommandobject($cmd) {
$path = "{$this->cmddir}/{$cmd}.php";
if (!file_exists($path)) {
throw new exception("cannot find $path", self::cmdman_illegalclass_error);
}
require_once $path;
if (!class_exists($cmd)) {
throw new exception("class $cmd does not exist", self::cmdman_illegalclass_error);
}
$class = new reflectionclass($cmd);
if (!$class->issubclassof(new reflectionclass('command'))) {
throw new exception("$cmd is not a command", self::cmdman_illegalclass_error);
}
return $class->newinstance();
}
}
?>
通过传递 cmdman_illegalclass_error和 cmdman_general_error其中之一的参数给我们抛出的异常对象,我们就可以让客户代码区分不同类型的错误,并定义不同的处理策略。
index_php5_3.php
<?php // php 5
try {
$mgr = new commandmanager();
$cmd = $mgr->getcommandobject('realcommand');
$cmd->execute();
} catch (exception $e) {
if ($e->getcode() == commandmanager::cmdman_general_error) {
// no way of recovering
die($e->getmessage());
} else if ($e->getcode() == commandmanager::cmdman_illegalclass_error) {
error_log($e->getmessage());
print "attempting recovery/n";
// perhaps attempt to invoke a default command?
}
}
?>
我们也可以用另一种方法来实现这样的效果—从最根本的exception类中派生出代表不同类型异常的子类,再抛出和捕捉。
新闻热点
疑难解答