燕之庐网站建设 - 优质网站设计公司

Posts Tagged ‘oop’

php

2009/11/19

利用PHP的OOP特性实现数据保护(3)

Tags: ,

函数execute

最后,还需要构建函数execute(),函数execute()编译query并且使用DB对 象执行它,而DB对象在此是用于初始化DBQuery对象的。请注意在例4中,是怎样运用函数call_user_func_array()来得到编译后 的query的,而这样做的原因是,函数execute()要直到运行时,才能确定传递给它的参数数目。

例4:execute()函数

/**
*
* 执行当前query,并把占位符替换为所提供的参数。
*
* @param mixed $queryParams,… Query parameter
* @return resource A reference to the resource representing the executed query.
*/
public function execute($queryParams = ”)
{
//例如:SELECT * FROM table WHERE name=:1S AND type=:2I AND level=:3N
$args = func_get_args();

if ($this->stored_procedure) {
/* 调用函数compile以取得query */
$query = call_user_func_array(array($this, ‘compile’), $args);
} else {
/* 如果存储过程未被初始化,就把它作为标准query执行。*/
$query = $queryParams;
}

$this->result = $this->db->query($query);

return $this->result;
}

全部整合起来

为演示怎样使用query对象,下面构造了一个小例子,其将把DBQuery对象作为存储过程使用,并检查是否输入了正确的用户名与密码,请看例5:

例5:

require ‘mysql_db.php5′;
require_once ‘query2.php5′;

$db = new MySqlDb;
$db->connect(’host’, ‘username’, ‘pass’);
$db->query(’use content_management_system’);

$query = new DBQuery($db);

$query->prepare(’SELECT fname,sname FROM users WHERE username=:1S AND pword=:2S AND expire_time<:3I’);

if ($result = $query->execute(”visualad”, “apron”, time())) {
if ($db->num_rows($result) == 1) {
echo(’凭证正确。’);
} else {
echo(’凭证不正确,会话已过期。’);
}
} else {
echo(’执行query时发生错误:’ . $db->error());
}

在本文中,你已看到了如何在声明类变量时,利用访问修饰符private、protected和public,保护数据和限制数据对象的可见性,同时,在PHP 5中,这些概念也可用于其他的数据类,保护其重要的内部数据。

php

利用PHP的OOP特性实现数据保护(2)

Tags: , ,

函数prepare

为使用例1中的模板,你要做的第一件事是构建好prepare()函数,为确保无带引号的字符被偶 然解析为占位符,函数应该移除query内所有字符串,并把它们临时存储在一个数组内。而字符串本身也会被占位符取代,其通常被识别为不应该在SQL语句 中出现的的字符串序列。在query的编译期间,过程占位符会首先被替换,接着把字符串放回query中,这是通过preg_replace()函数,和 另一个用作preg_replace()函数的helper回调函数完成的。

例2:prepare()函数

/**
* 把query准备为一个存储过程。
* @param string $query Prepared query text
* @return void
*/
public function prepare($query)
{
$this->stored_procedure = true;
$this->quote_store = array(); //清除引号
$this->query = preg_replace(self::$QUOTE_MATCH, ‘$this->sql_quote_replace(”1″?”1″:’2′)’, $query);
}

private function sql_quote_replace($match)
{
$number = count($this->query_strings);
$this->query_strings[] = $match;
return “$||$$number”;
}

在此留意对静态QUOTE_MATCH属性private的使用,还有quote_store属性和sql_quote_replace()函数。相比 protected,在此定义为private更能确保任何重载query类prepare()方法的子类使用其自身的机制来剔除引号。

函数compile

下一步是构建compile()与execute()函数。

函数compile()如例3中所示,功能如下:

·接受的参数数目可变(即可变参数),其将匹配query中的占位符。

·检查占位符是否为正确的数据类型,并把它替换为参数中的值。

·把query作为字符串返回,但不执行它。

·如果query对象没有使用prepare()函数初始化为一个存储过程,将抛出一个异常。

例3:compile()函数

/**
* 返回编译的query,但并不执行它。
* @param mixed $args,… Query Parameters
* @return string Compiled Query
*/
public function compile($params)
{
if (! $this->stored_procedure) {
throw new Exception(”存储过程未被初始化!”);
}

/* 替代参数 */
$params = func_get_args(); // 取得函数参数
$query = preg_replace(”/(?query);

return $this->add_strings($query); //把字符串放回query中
}

/**
* 重新插入被prepare()函数移除的字符串。
*/
private function add_strings($string)
{
$numbers = array_keys($this->query_strings);
$count = count($numbers);
$searches = array();
for($x = 0; $x < $count; $x++) {
$searches[$x] = “$||${$numbers[$x]}”;
}

return str_replace($searches, $this->query_strings, $string);
}

/**
* 每次执行,存储过程中都有一个占位符被替换。
*/
protected function compile_callback($params, $index, $type)
{
–$index;

/* 抛出一个异常 */
if (! isset($params[$index])) {
throw new Exception(”存储过程未收到所需的参数数目!”);
}

/* 可以在此添加别的类型,如日期和时间。 */
switch ($type) {
case ‘S’:
return ‘”‘ . $this->db->escape_string($params[$index]) . ‘”‘;
break;
case ‘I’:
return (int) $params[$index];
break;
case ‘N’:
return (float) $params[$index];
default:
throw new Exception(”存储过程中指定的数据类型 ‘$type’ 无法识别。”);
}
}

函数compile()中使用了两个额外的函数,其中compile_callback()函数是作为在preg_replace()函数调用中的回调函数,每一次在query中查找到占位符,并把它替换为传给compile函数的值时,都会执行它。

php

利用PHP的OOP特性实现数据保护(1 )

Tags: , ,

PHP 4中,声明变量通常使用var,而在PHP 5中,可使用面向对象编程(OOP)的特性来自定义数据的可见性–即可访问性,可见性在此与变量作用域非常类似,但提供了更好的控制机制,有以下三种类型的可见性修饰符:

Public(默认)–变量可在全局范围内访问或修改。
Protected–变量只能在类本身及直接派生(使用extends语句)类内访问或修改。
Private–变量只能在类内部访问或修改。

与接口实现类似,在程序中违反这些规则将会导致严重的错误;且与接口类似的是,它们的存在纯粹是为了方便程序员。但这并不意味着可以忽略它们,指定某个类成员变量的可见性,可保护对象内的数据免受外界影响。

假设有一个MySqlDB类,一个$link变量在其中声明为private,这意味着这个变量只能从对象内部使用$this变量访问,这防止了类外其他对象或函数的意外覆盖,在此,我们将使用可见性特性帮助我们创建一个query对象。

你可以把query当作一个单独的实体,它可以执行,并且返回结果。一些数据库系统也具有存储过程,存储过程与函数很相似,它们存储查询语句,并在调用时接受相应的参数,但MySQL在5.1版本之前并没有提供类似功能,某些其他类型的数据库管理系统也没有。

在本文中,将把上述两个特性结合进示例的query对象中,示例将模拟一个基本的存储过程,并在内部保存结果指针。目前,重点是从对象中执行query,在此可以调用MySqlDB对象的query()函数。

可在query对象中定义如下的public函数:

__construct()–构造函数接受一个包含了实现DB接口对象实例引用的参数。

prepare()–函数prepare()初始化query的存储过程。它可能包含一个或多个有限的占位符,而其将会作为参数传递给execute()函数。占位符定义为与参数个数有关的一个冒号紧跟一个整数及与参数类型有关的一个字母。

包含占位符的一个简单的query看起来像以下这样:

SELECT col1,col2 FROM table_name WHERE col1=:1I

execute()–函数execute()将执行query。如果它被prepare()函数过早地初始化为一个存储过程,任何传递进来的参数都会被作为存储过程的执行参数,否则,第一个参数只会被作为查询文本。函数execute()将返回执行查询后的结果。

compile()–函数compile()与函数execute()类似,实际上,query并没有执行,而是替换查询字符串中所有占位符,接受存储过程的参数,并返回query的编译版本。

受保护的成员

正如上面所提到的,可见性的概念可用于隐藏对象的内部工作,保护内部工作所需的数据完整性。前面已经解释,query返回的结果指针将会保存为protected属性,在此使用保护成员是因为从query对象派生出来的特定数据库query对象可能会重载某些核心功能。

深掘代码

理论说够了,现在开始编写代码,首先,创建一个例1所示的模板:

例1:数据库query类的一个模板

class DBQuery
{
/**
*保存一个实现了DB接口对象的引用。
*/
protected $db;

/**
*如果是一个存储过程,设为true。
*/
protected $stored_procedure = false;

/**
*保存一个删除了所有字符串的query。
*/
private $query;

/**
*用于在SQL中匹配引号。
*/
private static $QUOTE_MATCH = “/(”.*(?db = $db;
}

public function prepare($query)
{
$this->stored_procedure = true;
}

public function compile($args)
{}

public function execute($query)
{}
}

php

PHP5 OOP编程之代理与定制异常(2)

Tags: , ,

三、 抛出异常

你可能已经从上面的代码中注意到,你捕获的是一个称为QueryException(我们将在后面实现这个对象)的异常。一个异常类似于一个错误,然而却更具有一般性。描述一个异常的最好的方法是使用emergency。尽管一个emergency可以不会是“致命的”,但是还是必须处理它。当在PHP中抛出一个异常时,执行的当前范围很快地被终止,不管它是一个函数,try..catch块还是脚本本身。然后,该异常遍历调用栈—终止每个执行范围,直到或者在一个try..catch块中捕获它或者它到达调用栈的顶部—此时它将生成一个致命错误。

异常处理是PHP 5中的另外一个新特征,当与OOP联用时,它能够实现良好地控制错误处理和报告。一个try..catch块是一种处理异常的重要机制。一旦被捕获,脚本将会从异常被捕获和被处理的代码的下一行继续执行。

如果查询失败,你需要改变你的execute函数以抛出一个异常。你将抛出一个称为QueryException的定制异常对象—导致错误的DBQuery对象被传递给它。

列表3.抛出一个异常。

/**
*执行当前查询
*
* 执行当前查询—用提供的参数代替任何点位符
* .
*
* @参数: mixed $queryParams,… 查询参数
* @返回:资源A—参考描述执行查询的资源。
*/
public function execute($queryParams = ”)
{
//例如: SELECT * FROM table WHERE name=:1S AND type=:2I AND level=:3N
$args = func_get_args();
if ($this->stored_procedure) {
/*调用compile函数以得到查询*/
$query = call_user_func_array(array($this, ‘compile’), $args);
} else {
/*一个存储过程没被初始化,因此,作为一种标准查询来执行之*/
$query = $queryParams;
}
$result = $this->db->query($query);
if (! $result) {
throw new QueryException($this);
}
$this->result = $result;
/* 注意现在我们怎么返回对象本身,这使我们能够从这个函数的返回结果中调用成员函数
*/
return $this;
}

四、 使用继承抛出定制异常

在PHP中,你可以抛出任何对象作为一个异常;但是,首先该异常应该继承自PHP的内置异常类。通过创建你自己的定制异常,你可以记录其它有关于该错误的信息,例如在一个日志文件中创建一个入口,或做你喜欢做的任何事情。你的定制异常将要做如下几件事情:

· 记录由查询产生的来自DB对象的错误消息。

· 给出查询错误发生所在行代码的准确细节—通过检查调用栈。

· 显示错误消息和查询文本—当被转换成一个字符串时。

为了得到错误信息和查询文本,需要对DBQuery对象作多处更改。

1. 一个新的protected属性—compiledQuery—需要被添加到类中。

2. compile()函数使用查询文本更新查询compiledQuery属性。

3. 应该加入一个检索编译的查询文本的函数。

4. 还应该加入一个函数—它得到当前的与DBQuery对象相关联的DB对象。

列表4.抛出一个异常。

class DBQuery
{
/**
*在调用compile()或execute()之后存储查询的编译版本
*
* @var string $compiledQuery
*/
protected $compiledQuery;
/**
* 返回编译的查询而不执行它。
* @参数:mixed $params,…查询参数
* @返回:字符串—编译的查询
*/
public function compile($params=”)
{
if (! $this->stored_procedure) {
throw new Exception(”存储过程没被初始化.”);
}
/*代替参数*/
$params = func_get_args(); //得到函数参数
$query = preg_replace(”/(?compile_callback($params, 1, “2″)’, $this->query);
return ($this->compiledQuery = $this->add_strings($query)); //把字符串放回查询中
}
public function getDB()
{
return $this->db;
}
public function getCompiledQuery()
{
return $this->compiledQuery;
}
}

现在,你可以实现QueryException类。注意你是如何遍历调用栈以在脚本中查找实际导致错误的位置的。这正好适用于当抛出异常的DBQuery对象是一个继承自DBQuery对象的子类的情况。

列表5:QueryException类。

/**
*查询异常
*
*当试图执行一个查询时,如果一个错误发生,将由{@link DBQuery}对象抛出错误
*/
class QueryException extends Exception
{
/**
* 查询文本
*
* @var字符串$QueryText;
*/
protected $QueryText;
/**
*来自数据库的错误号/代码
*
* @var字符串$ErrorCode
*/
protected $ErrorNumber;
/**
*来自数据库的错误消息
*
* @var字符串$ErrorMessage
*/
protected $ErrorMessage;
/**
*类构造器
*
* @参数:DBQuery $db,是抛出异常的查询对象
*/
public function __construct(DBQuery $query)
{
/*得到调用栈*/
$backtrace = $this->GetTrace();
/*把行和文件设置到错误实际发生的位置*/
if (count($backtrace) > 0) {
$x = 1;
/*如果查询类被继承,那么我们需要忽略由子类所进行的调用*/
while((! isset($backtrace[$x]['line'])) ||
(isset($backtrace[$x]['class']) && is_subclass_of($backtrace[$x]['class'], ‘DBQuery’)) ||
(strpos(strtolower(@$backtrace[$x]['function']), ‘call_user_func’)) !== false ) {
/*循环执行,只要没有行号或调用的函数是DBQuery类的一个子类*/
++$x;
/*如果我们到达栈底,那么我们使用第一个调用者*/
if (($x) >= count($backtrace)) {
$x = count($backtrace);
break;
}
}
/*如果上面的循环至少执行一次,那么我们可以把它减1以查找实际的引起错误的代码行
*/
if ($x != 1) {
$x -= 1;
}
/*最后,我们可以设置文件和行号,这应该可以反映出引起错误的SQL语句*/
$this->line = $backtrace[$x]['line'];
$this->file = $backtrace[$x]['file'];
}
$this->QueryText = $query->getCompiledQuery();
$this->ErrorNumber = $query->getDB()->errno();
$this->ErrorMessage = $query->getDB()->error();
/*调用超类的异常构造器*/
parent::__construct(’Query Error’, 0);
}
/**
*得到查询文本
*
* @返回字符串查询文本
*/
public function GetQueryText()
{
return $this->QueryText;
}
/**
*得到错误号
*
* @返回字符串错误号
*/
public function GetErrorNumber()
{
return $this->ErrorNumber;
}
/**
*得到错误消息
*
* @返回字符串错误消息
*/
public function GetErrorMessage()
{
return $this->ErrorMessage;
}
/**
*当对象被转换为一个字符串时调用。
* @返回字符串
*/
public function __toString()
{
$output = “Query Error in {$this->file} on line {$this->line}nn”;
$output .= “Query: {$this->QueryText}n”;
$output .= “Error: {$this->ErrorMessage} ({$this->ErrorNumber})nn”;

return $output;
}
}

至此,在本节开始看到的代码可以工作了。

五、 结论

在本文中,你看到了代理是怎样把与查询相联系的DB接口映射到针对一个特定的查询结果上的操作。DBQuery对象暴露相同的函数,例如 fetch_assoc(),作为DB对象。然而,这些都是针对单个查询起作用。你还学习了如何使用定制异常来给出详细信息—一个错误发生在何时何地,以 及它们怎样更好地控制错误的处理。

php

PHP5 OOP编程之代理与定制异常(1)

Tags: , ,

一、 DBQuery对象

现在,我们的DBQuery对象简单地模仿一个存储过程—一旦被执 行,即返回一个必须进行保存的结果资源;并且如果你想使用该结果集上的函数(例如num_rows()或fetch_row())的话,你必须传递 MySqlDB对象。那么,如果由DBQuery对象来实现MySqlDB对象(其设计目的是对一个执行查询的结果进行操作)实现的函数,效果如何呢?让 我们继续使用上一篇示例中的代码;并且让我们假定,现在由DBQuery对象管理我们的结果资源。DBQuery类的源码如列表1所示。

列表1.使用DBQuery类。

require ‘mysql_db.php’;
require_once ‘query.php’;
$db = new MySqlDb;
$db->connect(’host’, ‘username’, ‘pass’);
$db->query(’use content_management_system’);
$query = new DBQuery($db);
$query->prepare(’SELECT fname,sname FROM users WHERE username=:1S AND pword=:2S AND expire_time<:3I’);
try {
if($query->execute(”visualad”, “apron”, time()))->num_rows() == 1) {
echo(’Correct Credentials’);
} else {
echo(’Incorrect Credentials / Session Expired’);
}
} catch (QueryException $e) {
echo(’Error executing query: ‘ . $e);
}

上面修改后的代码中我们最感兴趣的是,catch语句和execute语句。

· execute语句不再返回一个结果资源,现在它返回DBQuery对象本身。

· DBQuery对象现在实现num_rows()函数—我们从DB接口中已经熟悉。

· 如果查询执行失败,它抛出一个QueryException类型的异常。当被转换成一个字符串时,它将返回发生的错误的细节信息。

为此,你需要使用代理。事实上,你在我们的DBQuery对象中已经使用代理了,但是现在将更为深入地使用它来把它与MySqlDB对象紧密绑定。该DBQuery对象已经被使用一个实现DB接口的对象初始化,并且它已经包含一个成员函数execute—由它调用DB对象的query()方法来执行该查询。这个DBQuery对象本身并不实际地查询数据库,它把这项任务交由DB对象来完成。这就是代理,其实是一个进程—借助于这个进程,通过把消息发送给另一个实现相同的或类似行为的对象,一个对象可以实现一个特别的行为。

为此,你需要修改DBQuery对象以便包括所有的函数—它们操作一个来自DB对象的结果资源。当执行查询以调用DB对象的相应函数并且返回它的结果时,你需要使用存储的结果。下列函数将被添加:

列表2:使用代理扩展DBQuery类。

class DBQuery
{
…..

public function fetch_array()
{
if (! is_resource($this->result)) {
throw new Exception(’Query not executed.’);
}
return $this->db->fetch_array($this->result);
}

public function fetch_row()
{
if (! is_resource($this->result)) {
throw new Exception(’Query not executed.’);
}
return $this->db->fetch_row($this->result);
}

public function fetch_assoc()
{
if (! is_resource($this->result)) {
throw new Exception(’Query not executed.’);
}
return $this->db->fetch_assoc($this->result);
}

public function fetch_object()
{
if (! is_resource($this->result)) {
throw new Exception(’Query not executed.’);
}
return $this->db->fetch_object($this->result);
}

public function num_rows()
{
if (! is_resource($this->result)) {
throw new Exception(’Query not executed.’);
}
return $this->db->num_rows($this->result);
}
}

每个函数的实现相当简单。它首先进行检查,以确保已经执行查询,然后把任务代理到DB对象,返回它的结果就好象它是查询对象本身(称作是基本数据库函数)一样。

二、 类型提示(Type Hinting)

为了使代理能够工作,我们需要确保DBQuery对象的$db变量是一个实现了DB接口的对象的实例。类型提示是PHP 5中的一种新特征,它能够使你把函数参数强制转换成特定类型的对象。在PHP 5之前,唯一的确保函数参数是一个特定对象类型的方法是使用PHP中所提供的类型检查函数(也即是is_a())。现在,你可以简单地强制转换对象类型— 通过在函数参数的前面加上类型名。你已经从我们的DBQuery对象中看到了类型提示,这样可以确保一个实现DB接口的对象被传递到对象构造器中。

public function __construct(DB $db)
{
$this->db = $db;
}

当使用类型提示时,你不仅可以指定对象类型,还可以指定抽象类和接口。

php

2009/10/24

PHP5 OOP编程之代理与定制异常

Tags: ,

现在,我们的DBQuery对象简单地模仿一个存储过程—一旦被执行,即返回一个必须进行保存的结果资源

一、 DBQuery对象

现在,我们的DBQuery对象简单地模仿一个存储过程—一旦被执行,即返回一个必须进行保存的结果资源;并且如果你想使用该结果集上的函数(例如 num_rows()或fetch_row())的话,你必须传递MySqlDB对象。那么,如果由DBQuery对象来实现MySqlDB对象(其设计 目的是对一个执行查询的结果进行操作)实现的函数,效果如何呢?让我们继续使用上一篇示例中的代码;并且让我们假定,现在由DBQuery对象管理我们的 结果资源。DBQuery类的源码如列表1所示。

列表1.使用DBQuery类。

require ‘mysql_db.php’;
require_once ‘query.php’;
$db = new MySqlDb;
$db->connect(’host’, ‘username’, ‘pass’);
$db->query(’use content_management_system’);
$query = new DBQuery($db);
$query->prepare(’SELECT fname,sname FROM users WHERE username=:1S AND pword=:2S AND expire_time<:3I’);
try {
if($query->execute(”visualad”, “apron”, time()))->num_rows() == 1) {
echo(’Correct Credentials’);
} else {
echo(’Incorrect Credentials / Session Expired’);
}
} catch (QueryException $e) {
echo(’Error executing query: ‘ . $e);
}

上面修改后的代码中我们最感兴趣的是,catch语句和execute语句。

· execute语句不再返回一个结果资源,现在它返回DBQuery对象本身。

· DBQuery对象现在实现num_rows()函数—我们从DB接口中已经熟悉。

· 如果查询执行失败,它抛出一个QueryException类型的异常。当被转换成一个字符串时,它将返回发生的错误的细节信息。

为此,你需要使用代理。事实上,你在我们的DBQuery对象中已经使用代理了,但是现在将更为深入地使用它来把它与MySqlDB对象紧密绑定。该 DBQuery对象已经被使用一个实现DB接口的对象初始化,并且它已经包含一个成员函数execute—由它调用DB对象的query()方法来执行该 查询。这个DBQuery对象本身并不实际地查询数据库,它把这项任务交由DB对象来完成。这就是代理,其实是一个进程—借助于这个进程,通过把消息发送 给另一个实现相同的或类似行为的对象,一个对象可以实现一个特别的行为。

为此,你需要修改DBQuery对象以便包括所有的函数—它们操作一个来自DB对象的结果资源。当执行查询以调用DB对象的相应函数并且返回它的结果时,你需要使用存储的结果。下列函数将被添加:

列表2:使用代理扩展DBQuery类。

class DBQuery
{
…..

public function fetch_array()
{
if (! is_resource($this->result)) {
throw new Exception(’Query not executed.’);
}
return $this->db->fetch_array($this->result);
}

public function fetch_row()
{
if (! is_resource($this->result)) {
throw new Exception(’Query not executed.’);
}
return $this->db->fetch_row($this->result);
}

public function fetch_assoc()
{
if (! is_resource($this->result)) {
throw new Exception(’Query not executed.’);
}
return $this->db->fetch_assoc($this->result);
}

public function fetch_object()
{
if (! is_resource($this->result)) {
throw new Exception(’Query not executed.’);
}
return $this->db->fetch_object($this->result);
}

public function num_rows()
{
if (! is_resource($this->result)) {
throw new Exception(’Query not executed.’);
}
return $this->db->num_rows($this->result);
}
}

每个函数的实现相当简单。它首先进行检查,以确保已经执行查询,然后把任务代理到DB对象,返回它的结果就好象它是查询对象本身(称作是基本数据库函数)一样。

二、 类型提示(Type Hinting)

为了使代理能够工作,我们需要确保DBQuery对象的$db变量是一个实现了DB接口的对象的实例。类型提示是PHP 5中的一种新特征,它能够使你把函数参数强制转换成特定类型的对象。在PHP 5之前,唯一的确保函数参数是一个特定对象类型的方法是使用PHP中所提供的类型检查函数(也即是is_a())。现在,你可以简单地强制转换对象类型— 通过在函数参数的前面加上类型名。你已经从我们的DBQuery对象中看到了类型提示,这样可以确保一个实现DB接口的对象被传递到对象构造器中。

public function __construct(DB $db)
{
$this->db = $db;
}

当使用类型提示时,你不仅可以指定对象类型,还可以指定抽象类和接口。
三、 抛出异常

你可能已经从上面的代码中注意到,你捕获的是一个称为QueryException(我们将在后面实现这个对象)的异常。一个异常类似于一个错误,然而 却更具有一般性。描述一个异常的最好的方法是使用emergency。尽管一个emergency可以不会是“致命的”,但是还是必须处理它。当在PHP 中抛出一个异常时,执行的当前范围很快地被终止,不管它是一个函数,try..catch块还是脚本本身。然后,该异常遍历调用栈—终止每个执行范围,直 到或者在一个try..catch块中捕获它或者它到达调用栈的顶部—此时它将生成一个致命错误。

异常处理是PHP 5中的另外一个新特征,当与OOP联用时,它能够实现良好地控制错误处理和报告。一个try..catch块是一种处理异常的重要机制。一旦被捕获,脚本将会从异常被捕获和被处理的代码的下一行继续执行。

如果查询失败,你需要改变你的execute函数以抛出一个异常。你将抛出一个称为QueryException的定制异常对象—导致错误的DBQuery对象被传递给它。

列表3.抛出一个异常。

/**
*执行当前查询
*
* 执行当前查询—用提供的参数代替任何点位符
* .
*
* @参数: mixed $queryParams,… 查询参数
* @返回:资源A—参考描述执行查询的资源。
*/
public function execute($queryParams = ”)
{
//例如: SELECT * FROM table WHERE name=:1S AND type=:2I AND level=:3N
$args = func_get_args();
if ($this->stored_procedure) {
/*调用compile函数以得到查询*/
$query = call_user_func_array(array($this, ‘compile’), $args);
} else {
/*一个存储过程没被初始化,因此,作为一种标准查询来执行之*/
$query = $queryParams;
}
$result = $this->db->query($query);
if (! $result) {
throw new QueryException($this);
}
$this->result = $result;
/* 注意现在我们怎么返回对象本身,这使我们能够从这个函数的返回结果中调用成员函数
*/
return $this;
}

四、 使用继承抛出定制异常

在PHP中,你可以抛出任何对象作为一个异常;但是,首先该异常应该继承自PHP的内置异常类。通过创建你自己的定制异常,你可以记录其它有关于该错误的信息,例如在一个日志文件中创建一个入口,或做你喜欢做的任何事情。你的定制异常将要做如下几件事情:

· 记录由查询产生的来自DB对象的错误消息。

· 给出查询错误发生所在行代码的准确细节—通过检查调用栈。

· 显示错误消息和查询文本—当被转换成一个字符串时。

为了得到错误信息和查询文本,需要对DBQuery对象作多处更改。

1. 一个新的protected属性—compiledQuery—需要被添加到类中。

2. compile()函数使用查询文本更新查询compiledQuery属性。

3. 应该加入一个检索编译的查询文本的函数。

4. 还应该加入一个函数—它得到当前的与DBQuery对象相关联的DB对象。

列表4.抛出一个异常。

class DBQuery
{
/**
*在调用compile()或execute()之后存储查询的编译版本
*
* @var string $compiledQuery
*/
protected $compiledQuery;
/**
* 返回编译的查询而不执行它。
* @参数:mixed $params,…查询参数
* @返回:字符串—编译的查询
*/
public function compile($params=”)
{
if (! $this->stored_procedure) {
throw new Exception(”存储过程没被初始化.”);
}
/*代替参数*/
$params = func_get_args(); //得到函数参数
$query = preg_replace(”/(?compile_callback($params, 1, “2″)’, $this->query);
return ($this->compiledQuery = $this->add_strings($query)); //把字符串放回查询中
}
public function getDB()
{
return $this->db;
}
public function getCompiledQuery()
{
return $this->compiledQuery;
}
}

现在,你可以实现QueryException类。注意你是如何遍历调用栈以在脚本中查找实际导致错误的位置的。这正好适用于当抛出异常的DBQuery对象是一个继承自DBQuery对象的子类的情况。

列表5:QueryException类。

/**
*查询异常
*
*当试图执行一个查询时,如果一个错误发生,将由{@link DBQuery}对象抛出错误
*/
class QueryException extends Exception
{
/**
* 查询文本
*
* @var字符串$QueryText;
*/
protected $QueryText;
/**
*来自数据库的错误号/代码
*
* @var字符串$ErrorCode
*/
protected $ErrorNumber;
/**
*来自数据库的错误消息
*
* @var字符串$ErrorMessage
*/
protected $ErrorMessage;
/**
*类构造器
*
* @参数:DBQuery $db,是抛出异常的查询对象
*/
public function __construct(DBQuery $query)
{
/*得到调用栈*/
$backtrace = $this->GetTrace();
/*把行和文件设置到错误实际发生的位置*/
if (count($backtrace) > 0) {
$x = 1;
/*如果查询类被继承,那么我们需要忽略由子类所进行的调用*/
while((! isset($backtrace[$x]['line'])) ||
(isset($backtrace[$x]['class']) && is_subclass_of($backtrace[$x]['class'], ‘DBQuery’)) ||
(strpos(strtolower(@$backtrace[$x]['function']), ‘call_user_func’)) !== false ) {
/*循环执行,只要没有行号或调用的函数是DBQuery类的一个子类*/
++$x;
/*如果我们到达栈底,那么我们使用第一个调用者*/
if (($x) >= count($backtrace)) {
$x = count($backtrace);
break;
}
}
/*如果上面的循环至少执行一次,那么我们可以把它减1以查找实际的引起错误的代码行
*/
if ($x != 1) {
$x -= 1;
}
/*最后,我们可以设置文件和行号,这应该可以反映出引起错误的SQL语句*/
$this->line = $backtrace[$x]['line'];
$this->file = $backtrace[$x]['file'];
}
$this->QueryText = $query->getCompiledQuery();
$this->ErrorNumber = $query->getDB()->errno();
$this->ErrorMessage = $query->getDB()->error();
/*调用超类的异常构造器*/
parent::__construct(’Query Error’, 0);
}
/**
*得到查询文本
*
* @返回字符串查询文本
*/
public function GetQueryText()
{
return $this->QueryText;
}
/**
*得到错误号
*
* @返回字符串错误号
*/
public function GetErrorNumber()
{
return $this->ErrorNumber;
}
/**
*得到错误消息
*
* @返回字符串错误消息
*/
public function GetErrorMessage()
{
return $this->ErrorMessage;
}
/**
*当对象被转换为一个字符串时调用。
* @返回字符串
*/
public function __toString()
{
$output = “Query Error in {$this->file} on line {$this->line}nn”;
$output .= “Query: {$this->QueryText}n”;
$output .= “Error: {$this->ErrorMessage} ({$this->ErrorNumber})nn”;

return $output;
}
}

至此,在本节开始看到的代码可以工作了。

五、 结论

在本文中,你看到了代理是怎样把与查询相联系的DB接口映射到针对一个特定的查询结果上的操作。DBQuery对象暴露相同的函数,例如 fetch_assoc(),作为DB对象。然而,这些都是针对单个查询起作用。你还学习了如何使用定制异常来给出详细信息—一个错误发生在何时何地,以 及它们怎样更好地控制错误的处理。