日韩天天综合网_野战两个奶头被亲到高潮_亚洲日韩欧美精品综合_av女人天堂污污污_视频一区**字幕无弹窗_国产亚洲欧美小视频_国内性爱精品在线免费视频_国产一级电影在线播放_日韩欧美内地福利_亚洲一二三不卡片区

《PHP設(shè)計模式介紹》第十五章 表數(shù)據(jù)網(wǎng)關(guān)模式(3)_PHP教程

編輯Tag賺U幣
教程Tag:暫無Tag,歡迎添加,賺取U幣!

推薦:《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 {
// ...
function testFindByTag() {
$gateway = new BookmarkGateway(DB::conn());
$this->addSeveralBookmarks($gateway);
$result = $gateway->findByTag(‘php’);
$this->assertIsA($result, ‘AdoResultSetIteratorDecorator’);
$count=0;
foreach($result as $bookmark) {
$count;
$this->assertIsA($bookmark, ‘ADOFetchObj’);
}
$this->assertEqual(3, $count);
}
}

findByTag方法的實現(xiàn)如下

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’;
class AdoResultSetIteratorDecorator implements Iterator {
protected $rs;
public function __construct($rs) {
$this->rs = new ADODB_Iterator($rs);
}
public function current() {
return $this->rs->fetchObj();
}
public function next() {
return $this->rs->next();
}
public function key() {
return $this->rs->key();
}
public function valid() {
return $this->rs->valid();
}
public function rewind() {
return $this->rs->rewind();
}
}

上述代碼中,大多數(shù)迭代器接口方法已作為代理來處理結(jié)果集了。但是current()方法被重載用于返回fetchObj()方法的結(jié)果。

分享:《PHP設(shè)計模式介紹》第十三章 適配器模式
接口的改變,是一個需要程序員們必須(雖然很不情愿)接受和處理的普遍問題。程序提供者們修改他們的代碼;系統(tǒng)庫被修正;各種程序語言以及相關(guān)庫的發(fā)展和進化。我孩子的無數(shù)玩具中有一個簡要地描

來源:模板無憂//所屬分類:PHP教程/更新時間:2008-08-22
相關(guān)PHP教程