淺談PHP5 OOP編程之代理與定制異常(2)_PHP教程
推薦:淺談PHP5 OOP編程之代理與定制異常(1)一、 DBQuery對(duì)象 現(xiàn)在,我們的DBQuery對(duì)象簡(jiǎn)單地模仿一個(gè)存儲(chǔ)過(guò)程一旦被執(zhí)行,即返回一個(gè)必須進(jìn)行保存的結(jié)果資源;并且如果你想使用該結(jié)果集上的函數(shù)(例如num_rows()或fetch_row())的話,你必須傳遞MySqlDB對(duì)象。那么,如果由DBQuery對(duì)象來(lái)實(shí)現(xiàn)MySqlDB對(duì)
三、 拋出異常
你可能已經(jīng)從上面的代碼中注意到,你捕獲的是一個(gè)稱(chēng)為QueryException(我們將在后面實(shí)現(xiàn)這個(gè)對(duì)象)的異常。一個(gè)異常類(lèi)似于一個(gè)錯(cuò)誤,然而卻更具有一般性。描述一個(gè)異常的最好的方法是使用emergency。盡管一個(gè)emergency可以不會(huì)是“致命的”,但是還是必須處理它。當(dāng)在PHP中拋出一個(gè)異常時(shí),執(zhí)行的當(dāng)前范圍很快地被終止,不管它是一個(gè)函數(shù),try..catch塊還是腳本本身。然后,該異常遍歷調(diào)用棧—終止每個(gè)執(zhí)行范圍,直到或者在一個(gè)try..catch塊中捕獲它或者它到達(dá)調(diào)用棧的頂部—此時(shí)它將生成一個(gè)致命錯(cuò)誤。
異常處理是PHP 5中的另外一個(gè)新特征,當(dāng)與OOP聯(lián)用時(shí),它能夠?qū)崿F(xiàn)良好地控制錯(cuò)誤處理和報(bào)告。一個(gè)try..catch塊是一種處理異常的重要機(jī)制。一旦被捕獲,腳本將會(huì)從異常被捕獲和被處理的代碼的下一行繼續(xù)執(zhí)行。
如果查詢(xún)失敗,你需要改變你的execute函數(shù)以拋出一個(gè)異常。你將拋出一個(gè)稱(chēng)為QueryException的定制異常對(duì)象—導(dǎo)致錯(cuò)誤的DBQuery對(duì)象被傳遞給它。
列表3.拋出一個(gè)異常。
/**
*執(zhí)行當(dāng)前查詢(xún)
*
* 執(zhí)行當(dāng)前查詢(xún)—用提供的參數(shù)代替任何點(diǎn)位符
* .
*
* @參數(shù): mixed $queryParams,... 查詢(xún)參數(shù)
* @返回:資源A—參考描述執(zhí)行查詢(xún)的資源。
*/
public function execute($queryParams = '')
{
//例如: SELECT * FROM table WHERE name=:1S AND type=:2I AND level=:3N
$args = func_get_args();
if ($this->stored_procedure) {
/*調(diào)用compile函數(shù)以得到查詢(xún)*/
$query = call_user_func_array(array($this, 'compile'), $args);
} else {
/*一個(gè)存儲(chǔ)過(guò)程沒(méi)被初始化,因此,作為一種標(biāo)準(zhǔn)查詢(xún)來(lái)執(zhí)行之*/
$query = $queryParams;
}
$result = $this->db->query($query);
if (! $result) {
throw new QueryException($this);
}
$this->result = $result;
/* 注意現(xiàn)在我們?cè)趺捶祷貙?duì)象本身,這使我們能夠從這個(gè)函數(shù)的返回結(jié)果中調(diào)用成員函數(shù)
*/
return $this;
}
四、 使用繼承拋出定制異常
在PHP中,你可以拋出任何對(duì)象作為一個(gè)異常;但是,首先該異常應(yīng)該繼承自PHP的內(nèi)置異常類(lèi)。通過(guò)創(chuàng)建你自己的定制異常,你可以記錄其它有關(guān)于該錯(cuò)誤的信息,例如在一個(gè)日志文件中創(chuàng)建一個(gè)入口,或做你喜歡做的任何事情。你的定制異常將要做如下幾件事情:
· 記錄由查詢(xún)產(chǎn)生的來(lái)自DB對(duì)象的錯(cuò)誤消息。
· 給出查詢(xún)錯(cuò)誤發(fā)生所在行代碼的準(zhǔn)確細(xì)節(jié)—通過(guò)檢查調(diào)用棧。
· 顯示錯(cuò)誤消息和查詢(xún)文本—當(dāng)被轉(zhuǎn)換成一個(gè)字符串時(shí)。
為了得到錯(cuò)誤信息和查詢(xún)文本,需要對(duì)DBQuery對(duì)象作多處更改。
1. 一個(gè)新的protected屬性—compiledQuery—需要被添加到類(lèi)中。
2. compile()函數(shù)使用查詢(xún)文本更新查詢(xún)compiledQuery屬性。
3. 應(yīng)該加入一個(gè)檢索編譯的查詢(xún)文本的函數(shù)。
4. 還應(yīng)該加入一個(gè)函數(shù)—它得到當(dāng)前的與DBQuery對(duì)象相關(guān)聯(lián)的DB對(duì)象。
列表4.拋出一個(gè)異常。
class DBQuery
{
/**
*在調(diào)用compile()或execute()之后存儲(chǔ)查詢(xún)的編譯版本
*
* @var string $compiledQuery
*/
protected $compiledQuery;
/**
* 返回編譯的查詢(xún)而不執(zhí)行它。
* @參數(shù):mixed $params,...查詢(xún)參數(shù)
* @返回:字符串—編譯的查詢(xún)
*/
public function compile($params='')
{
if (! $this->stored_procedure) {
throw new Exception("存儲(chǔ)過(guò)程沒(méi)被初始化.");
}
/*代替參數(shù)*/
$params = func_get_args(); //得到函數(shù)參數(shù)
$query = preg_replace("/(?compile_callback($params, 1, "2")', $this->query);
return ($this->compiledQuery = $this->add_strings($query)); //把字符串放回查詢(xún)中
}
public function getDB()
{
return $this->db;
}
public function getCompiledQuery()
{
return $this->compiledQuery;
}
}
現(xiàn)在,你可以實(shí)現(xiàn)QueryException類(lèi)。注意你是如何遍歷調(diào)用棧以在腳本中查找實(shí)際導(dǎo)致錯(cuò)誤的位置的。這正好適用于當(dāng)拋出異常的DBQuery對(duì)象是一個(gè)繼承自DBQuery對(duì)象的子類(lèi)的情況。
列表5:QueryException類(lèi)。
/**
*查詢(xún)異常
*
*當(dāng)試圖執(zhí)行一個(gè)查詢(xún)時(shí),如果一個(gè)錯(cuò)誤發(fā)生,將由{@link DBQuery}對(duì)象拋出錯(cuò)誤
*/
class QueryException extends Exception
{
/**
* 查詢(xún)文本
*
* @var字符串$QueryText;
*/
protected $QueryText;
/**
*來(lái)自數(shù)據(jù)庫(kù)的錯(cuò)誤號(hào)/代碼
*
* @var字符串$ErrorCode
*/
protected $ErrorNumber;
/**
*來(lái)自數(shù)據(jù)庫(kù)的錯(cuò)誤消息
*
* @var字符串$ErrorMessage
*/
protected $ErrorMessage;
/**
*類(lèi)構(gòu)造器
*
* @參數(shù):DBQuery $db,是拋出異常的查詢(xún)對(duì)象
*/
public function __construct(DBQuery $query)
{
/*得到調(diào)用棧*/
$backtrace = $this->GetTrace();
/*把行和文件設(shè)置到錯(cuò)誤實(shí)際發(fā)生的位置*/
if (count($backtrace) > 0) {
$x = 1;
/*如果查詢(xún)類(lèi)被繼承,那么我們需要忽略由子類(lèi)所進(jìn)行的調(diào)用*/
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 ) {
/*循環(huán)執(zhí)行,只要沒(méi)有行號(hào)或調(diào)用的函數(shù)是DBQuery類(lèi)的一個(gè)子類(lèi)*/
$x;
/*如果我們到達(dá)棧底,那么我們使用第一個(gè)調(diào)用者*/
if (($x) >= count($backtrace)) {
$x = count($backtrace);
break;
}
}
/*如果上面的循環(huán)至少執(zhí)行一次,那么我們可以把它減1以查找實(shí)際的引起錯(cuò)誤的代碼行
*/
if ($x != 1) {
$x -= 1;
}
/*最后,我們可以設(shè)置文件和行號(hào),這應(yīng)該可以反映出引起錯(cuò)誤的SQL語(yǔ)句*/
$this->line = $backtrace[$x]['line'];
$this->file = $backtrace[$x]['file'];
}
$this->QueryText = $query->getCompiledQuery();
$this->ErrorNumber = $query->getDB()->errno();
$this->ErrorMessage = $query->getDB()->error();
/*調(diào)用超類(lèi)的異常構(gòu)造器*/
parent::__construct('Query Error', 0);
}
/**
*得到查詢(xún)文本
*
* @返回字符串查詢(xún)文本
*/
public function GetQueryText()
{
return $this->QueryText;
}
/**
*得到錯(cuò)誤號(hào)
*
* @返回字符串錯(cuò)誤號(hào)
*/
public function GetErrorNumber()
{
return $this->ErrorNumber;
}
/**
*得到錯(cuò)誤消息
*
* @返回字符串錯(cuò)誤消息
*/
public function GetErrorMessage()
{
return $this->ErrorMessage;
}
/**
*當(dāng)對(duì)象被轉(zhuǎn)換為一個(gè)字符串時(shí)調(diào)用。
* @返回字符串
*/
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;
}
}
至此,在本節(jié)開(kāi)始看到的代碼可以工作了。
五、 結(jié)論
在本文中,你看到了代理是怎樣把與查詢(xún)相聯(lián)系的DB接口映射到針對(duì)一個(gè)特定的查詢(xún)結(jié)果上的操作。DBQuery對(duì)象暴露相同的函數(shù),例如fetch_assoc(),作為DB對(duì)象。然而,這些都是針對(duì)單個(gè)查詢(xún)起作用。你還學(xué)習(xí)了如何使用定制異常來(lái)給出詳細(xì)信息—一個(gè)錯(cuò)誤發(fā)生在何時(shí)何地,以及它們?cè)鯓痈玫乜刂棋e(cuò)誤的處理。
分享:如何利用PHP操縱Oracle LOB類(lèi)型數(shù)據(jù)用過(guò)Oracle的人都知道,Oracle有一種數(shù)據(jù)類(lèi)型叫VARCHAR2,用來(lái)表示不定長(zhǎng)的字符串。VARCHAR2也是Oracle公司推薦使用的類(lèi)型。但使用VARCHAR2有個(gè)問(wèn)題:最大只能表示4000個(gè)字符,也就相當(dāng)于2000個(gè)漢字。如果你的程序中某個(gè)字符的值要大于20002個(gè)漢字,用VARCHA
- PHPNOW安裝Memcached擴(kuò)展方法詳解
- php記錄頁(yè)面代碼執(zhí)行時(shí)間
- PHP中獎(jiǎng)概率的抽獎(jiǎng)算法程序代碼
- apache設(shè)置靜態(tài)文件緩存方法介紹
- php對(duì)圖像的各種處理函數(shù)代碼小結(jié)
- PHP 關(guān)于訪問(wèn)控制的和運(yùn)算符優(yōu)先級(jí)介紹
- 關(guān)于PHP語(yǔ)言構(gòu)造器介紹
- php/js獲取客戶端mac地址的實(shí)現(xiàn)代碼
- php5.5新數(shù)組函數(shù)array_column使用
- PHP preg_match的匹配多國(guó)語(yǔ)言的技巧
- php 中序列化和json使用介紹
- php采集文章中的圖片獲取替換到本地
- 相關(guān)鏈接:
- 教程說(shuō)明:
PHP教程-淺談PHP5 OOP編程之代理與定制異常(2)
。