《PHP設計模式介紹》第十五章 表數據網關模式(3)_PHP教程
推薦:《PHP設計模式介紹》第十四章 動態記錄模式到目前為止,您所看到的這些設計模式大大提高了代碼的可讀性與可維護性。然而,在WEB應用設計與開發中一個基本的需求與挑戰:數據庫應用,這些設計模式都沒有涉及到。本章與接下來的兩章—
ADOdb默認返回的是行記錄的散列數組,我有意讓這個例子稍微復雜一點,來強制其返回一個數據轉輸對象,這樣代碼就有意思多了。并且你將看到這個示例應用了前面學習過的一些設計模式)。
以下測試用例是上述需求的簡化表述。
|
class TableDataGatewayTestCase extends BaseTestCase { |
| class BookmarkGateway{ // ... public function findByTag($tag) { $rs = $this->conn->execute( ‘select * from bookmark where tag like ?’ ,array($tag.’%’)); return new AdoResultSetIteratorDecorator($rs); } } |
很典型的,findByTag()首先調用execute()方法生成一個數據集。ADOdb的execute()方法帶入兩個參數,待執行的SQL語句和一個可選的梆定參數變量的數組。因為findByTag()需要用帶通配符的LIKE操作,并且ADOdb會自動的給查詢字串加引號,所以必須要給作為參數的數組在其內部就加上通配符。Execute()產生一個記錄集后,AdoResultSetIteratorDecorator()將對其進行封包。AdoResultSetIteratorDecorator()的主要目的在于把結果集“轉換”為可迭代的對象集合,也因此而得名。
ADOdb通過包含adodb-iterator.inc.php提供對迭代的支持。其中定義了一個ADODB_Iterator的類,其實質是將ADOResultSet修飾成為PHP5的一個迭代接口標準庫。這使得你可以快速的形成一個可以遍歷的結果集了。然而,迭代器的默認行為還是返回一個聚合數組。正如你將在下述試驗中看到的那樣。
| class AdoResultSetIteratorDecoratorTestCase extends BaseTestCase { function testADOdbDecorator() { $gateway = new BookmarkGateway($this->conn); $this->addSeveralBookmarks($gateway); $rs = $this->conn->execute(‘select * from bookmark’); foreach($rs as $row) { $this->assertIsA($row, ‘array’); $this->assertIsA($rs->fetchObj(), ‘ADOFetchObj’); } } } |
這兒,通過ADOdb迭代器,表數據就可以被建立,存儲,迭代獲取數據。
突出顯示的代碼行實際是無效,要注意避免。你的確能為每一行生成一個對象,如果這樣,你就不得不在你的應用中到處重復這個笨拙的代碼來實現對整個集合的迭代。
一個更好的解決方案――能更直接的滿足對象集合迭代要求的是:修飾ADOdb迭代器。
測試外部庫
寫一個小測試用例來幫助你探測第三方庫,更好的了解它們的特點。一系列的測試用例也能使你更好的把握住對外部庫的依賴性(獨立性),或是你的代碼是如何特定的使用這些庫,這樣當庫因升級而改變時能更快的找到并解決問題。
如果你擔心對這些外部庫的依賴性,則引入適配器(見第十三章--適配器模式)使你的代碼從這種依賴關系中獨立出來。
讓我們寫一個測試用例來演示迭代器是如何工作的。
| class AdoResultSetIteratorDecoratorTestCase extends BaseTestCase { // ... function testRsDecorator() { $gateway = new BookmarkGateway($this->conn); $this->addSeveralBookmarks($gateway); $rs = $this->conn->execute(‘select * from bookmark’); $count=0; foreach(new AdoResultSetIteratorDecorator($rs) as $bookmark) { $count; $this->assertIsA($bookmark, ‘ADOFetchObj’); $this->assertTrue($bookmark->id > 0); $this->assertTrue(strlen($bookmark->url) > 10); } $this->assertEqual(5,$count); } } |
以下代碼說明了怎樣改進(修飾)ADODB_Iterator來滿足上述的需求。
|
require_once ‘adodb/adodb-iterator.inc.php’; |
上述代碼中,大多數迭代器接口方法已作為代理來處理結果集了。但是current()方法被重載用于返回fetchObj()方法的結果。
分享:《PHP設計模式介紹》第十三章 適配器模式接口的改變,是一個需要程序員們必須(雖然很不情愿)接受和處理的普遍問題。程序提供者們修改他們的代碼;系統庫被修正;各種程序語言以及相關庫的發展和進化。我孩子的無數玩具中有一個簡要地描
- 相關鏈接:
- 教程說明:
PHP教程-《PHP設計模式介紹》第十五章 表數據網關模式(3)
。