国产探花免费观看_亚洲丰满少妇自慰呻吟_97日韩有码在线_资源在线日韩欧美_一区二区精品毛片,辰东完美世界有声小说,欢乐颂第一季,yy玄幻小说排行榜完本

首頁 > 開發 > PHP > 正文

淺談PHP5 OOP編程之代理與定制異常(2)

2024-05-04 21:52:04
字體:
來源:轉載
供稿:網友

三、 拋出異常

你可能已經從上面的代碼中注意到,你捕獲的是一個稱為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對象。然而,這些都是針對單個查詢起作用。你還學習了如何使用定制異常來給出詳細信息—一個錯誤發生在何時何地,以及它們怎樣更好地控制錯誤的處理。

發表評論 共有條評論
用戶名: 密碼:
驗證碼: 匿名發表
主站蜘蛛池模板: 泗洪县| 石渠县| 株洲市| 陇川县| 逊克县| 马山县| 九江市| 陵川县| 红安县| 广州市| 景洪市| 潜山县| 梅河口市| 兰溪市| 常州市| 承德县| 木兰县| 墨玉县| 沂源县| 莲花县| 巴马| 项城市| 社旗县| 二手房| 桐乡市| 浪卡子县| 阿城市| 卫辉市| 屏南县| 获嘉县| 泗水县| 文昌市| 永州市| 三门县| 枞阳县| 高要市| 涿鹿县| 海安县| 晋州市| 隆林| 当阳市|