《PHP設(shè)計模式介紹》第十五章 表數(shù)據(jù)網(wǎng)關(guān)模式(3)_PHP教程
推薦:《PHP設(shè)計模式介紹》第十四章 動態(tài)記錄模式到目前為止,您所看到的這些設(shè)計模式大大提高了代碼的可讀性與可維護性。然而,在WEB應(yīng)用設(shè)計與開發(fā)中一個基本的需求與挑戰(zhàn):數(shù)據(jù)庫應(yīng)用,這些設(shè)計模式都沒有涉及到。本章與接下來的兩章—
ADOdb默認返回的是行記錄的散列數(shù)組,我有意讓這個例子稍微復(fù)雜一點,來強制其返回一個數(shù)據(jù)轉(zhuǎn)輸對象,這樣代碼就有意思多了。并且你將看到這個示例應(yīng)用了前面學(xué)習(xí)過的一些設(shè)計模式)。
以下測試用例是上述需求的簡化表述。
|
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()首先調(diào)用execute()方法生成一個數(shù)據(jù)集。ADOdb的execute()方法帶入兩個參數(shù),待執(zhí)行的SQL語句和一個可選的梆定參數(shù)變量的數(shù)組。因為findByTag()需要用帶通配符的LIKE操作,并且ADOdb會自動的給查詢字串加引號,所以必須要給作為參數(shù)的數(shù)組在其內(nèi)部就加上通配符。Execute()產(chǎn)生一個記錄集后,AdoResultSetIteratorDecorator()將對其進行封包。AdoResultSetIteratorDecorator()的主要目的在于把結(jié)果集“轉(zhuǎn)換”為可迭代的對象集合,也因此而得名。
ADOdb通過包含adodb-iterator.inc.php提供對迭代的支持。其中定義了一個ADODB_Iterator的類,其實質(zhì)是將ADOResultSet修飾成為PHP5的一個迭代接口標(biāo)準(zhǔn)庫。這使得你可以快速的形成一個可以遍歷的結(jié)果集了。然而,迭代器的默認行為還是返回一個聚合數(shù)組。正如你將在下述試驗中看到的那樣。
| 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迭代器,表數(shù)據(jù)就可以被建立,存儲,迭代獲取數(shù)據(jù)。
突出顯示的代碼行實際是無效,要注意避免。你的確能為每一行生成一個對象,如果這樣,你就不得不在你的應(yīng)用中到處重復(fù)這個笨拙的代碼來實現(xiàn)對整個集合的迭代。
一個更好的解決方案――能更直接的滿足對象集合迭代要求的是:修飾ADOdb迭代器。
測試外部庫
寫一個小測試用例來幫助你探測第三方庫,更好的了解它們的特點。一系列的測試用例也能使你更好的把握住對外部庫的依賴性(獨立性),或是你的代碼是如何特定的使用這些庫,這樣當(dāng)庫因升級而改變時能更快的找到并解決問題。
如果你擔(dān)心對這些外部庫的依賴性,則引入適配器(見第十三章--適配器模式)使你的代碼從這種依賴關(guān)系中獨立出來。
讓我們寫一個測試用例來演示迭代器是如何工作的。
| 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’; |
上述代碼中,大多數(shù)迭代器接口方法已作為代理來處理結(jié)果集了。但是current()方法被重載用于返回fetchObj()方法的結(jié)果。
分享:《PHP設(shè)計模式介紹》第十三章 適配器模式接口的改變,是一個需要程序員們必須(雖然很不情愿)接受和處理的普遍問題。程序提供者們修改他們的代碼;系統(tǒng)庫被修正;各種程序語言以及相關(guān)庫的發(fā)展和進化。我孩子的無數(shù)玩具中有一個簡要地描
- 相關(guān)鏈接:
復(fù)制本頁鏈接| 搜索《PHP設(shè)計模式介紹》第十五章 表數(shù)據(jù)網(wǎng)關(guān)模式(3)
- 教程說明:
PHP教程-《PHP設(shè)計模式介紹》第十五章 表數(shù)據(jù)網(wǎng)關(guān)模式(3)
。