首页 > 开发 > PHP > 正文

PHP 5.0异常处理机制深度探索

2024-05-04 22:56:18
字体:
来源:转载
供稿:网友
   本文面向希望了解php5异常处理机制的程序员。阅读本文你需要具有一定面向对象编程和php基础。

  php5内建的异常类需要有以下成员方法:

__construct()构造函数,需要一个出错信息和一个可选的整型错误标记作参数getmessage()取得出错信息 getcode()
出错的代码 getfile()异常发生的文件getline()异常发生的行数 gettrace()跟踪异常每一步传递的路线,存入数组,返回该数组 gettraceasstring()和gettrace()功能一样,但可以将数组中的元素转成字符串并按一定格式输出
  可以看出来,exception 类的结构和pear_error 很相似。当你的脚本中遇到一个错误,你可以建立你的异常对象:

$ex = new exception( "could not open $this->file" );
  exception类的构造函数将接受一个出错信息和一个错误代码。

  使用 throw关键字

  建立一个exception对象后你可以将对象返回,但不应该这样使用,更好的方法是用throw关键字来代替。throw用来抛出异常:

throw new exception( "my message", 44 );
  throw 将脚本的执行中止,并使相关的exception对象对客户代码可用。

  以下是改进过的getcommandobject() 方法:

  index_php5.php

<?php // php 5
require_once('cmd_php5/command.php');
class commandmanager {
 private $cmddir = "cmd_php5";

 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();
}
}
?>
  代码中我们使用了php5的反射(reflection)api来判断所给的类是否是属于command 类型。在错误的路径下执行本脚本将会报出这样的错误:

fatal error: uncaught exception 'exception' with message 'cannot find command/xrealcommand.php' in /home/xyz/basicexception.php:10
stack trace:
#0 /home/xyz/basicexception.php(26):
commandmanager->getcommandobject('xrealcommand')
#1 {main}
thrown in /home/xyz/basicexception.php on line 10
  默认地,抛出异常导致一个fatal error。这意味着使用异常的类内建有安全机制。而仅仅使用一个错误标记,不能拥有这样的功能。处理错误标记失败只会你的脚本使用错误的值来继续执行。

  try-catch 语句

  为了进一步处理异常,我们需要使用try-catch语句—包括try语句和至少一个的catch语句。任何调用 可能抛出异常的方法的代码都应该使用try语句。catch语句用来处理可能抛出的异常。以下显示了我们处理getcommandobject()抛出的异常的方法:

  index_php5.php 后半段

<?php
// php 5
try {
 $mgr = new commandmanager();
 $cmd = $mgr->getcommandobject('realcommand');
 $cmd->execute();
} catch (exception $e) {
 print $e->getmessage();
 exit();
}
?>


  可以看到,通过结合使用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类中派生出代表不同类型异常的子类,再抛出和捕捉。

发表评论 共有条评论
用户名: 密码:
验证码: 匿名发表