PHP設(shè)計(jì)模式之裝飾器模式定義與用法詳解
本文實(shí)例講述了PHP設(shè)計(jì)模式之裝飾器模式定義與用法。分享給大家供大家參考,具體如下:
什么是裝飾器模式
作為一種結(jié)構(gòu)型模式, 裝飾器(Decorator)模式就是對(duì)一個(gè)已有結(jié)構(gòu)增加"裝飾".
適配器模式, 是為現(xiàn)在有結(jié)構(gòu)增加的是一個(gè)適配器類,.將一個(gè)類的接口,轉(zhuǎn)換成客戶期望的另外一個(gè)接口.適配器讓原本接口不兼容的類可以很好的合作.
裝飾器模式是將一個(gè)對(duì)象包裝起來以增強(qiáng)新的行為和責(zé)任.裝飾器也稱為包裝器(類似于適配器)
有些設(shè)計(jì)設(shè)計(jì)模式包含一個(gè)抽象類,而且該抽象類還繼承了另一個(gè)抽象類,這種設(shè)計(jì)模式為數(shù)不多,而裝飾器就是其中之一.
什么時(shí)候使用裝飾器模式
基本說來, 如果想為現(xiàn)有對(duì)象增加新功能而不想影響其他對(duì)象, 就可以使用裝飾器模式.如果你好不容易為客戶創(chuàng)建了一個(gè)網(wǎng)站格式, 主要組件的工作都很完美, 客戶請(qǐng)求新功能時(shí), 你肯定不希望推翻重來, 再重新創(chuàng)建網(wǎng)站. 例如, 假設(shè)你已經(jīng)構(gòu)建了客戶原先請(qǐng)求的組件, 之后客戶又有了新的需求, 希望在網(wǎng)站中包含視頻功能. 你不用重寫原先的組件, 只需要"裝飾"現(xiàn)有組件, 為它們?cè)黾右曨l功能. 這樣即可以保持原來的功能,還可以增加新功能.
有些項(xiàng)目可能有時(shí)需要裝飾, 而有時(shí)不希望裝飾, 這些項(xiàng)目體現(xiàn)了裝飾器設(shè)計(jì)模式的另一個(gè)重要特性.假設(shè)你的基本網(wǎng)站開發(fā)模式可以滿足大多數(shù)客戶的要求. 不過, 胡些客戶還希望有一些特定的功能來滿足他們的需求. 并不是所有人都希望或需要這些額外的功能. 作為開發(fā)人員, 你希望你創(chuàng)建的網(wǎng)站能滿足客戶的業(yè)務(wù)目標(biāo). 所以需要提供"本地化"(customerization)特性, 即針對(duì)特定業(yè)務(wù)提供的特性. 利用裝飾器模式, 不僅能提供核心功能, 還可以用客戶要求的特有功能"裝飾"這些核心功能.
簡單的裝飾器例子
一個(gè)web開發(fā)企業(yè),計(jì)劃建立一個(gè)基本網(wǎng)站,并提供一些增強(qiáng)功能. 不過,web開發(fā)人員知道, 盡管這個(gè)基本計(jì)劃適用于大多數(shù)客戶, 但客戶以后很可能還希望進(jìn)一步提升, 利用裝飾器模式, 可以很容易地增加多個(gè)具體裝飾器,另外由于你能選擇要增加的裝飾器, 所以企業(yè)不僅能控制功能, 還可以控制項(xiàng)目的成本 .
Component接口
Component參與者是一個(gè)接口, 在這里, 它是一個(gè)抽象類IComponent. 這個(gè)抽象類只有一個(gè)屬性$site, 另外有兩個(gè)抽象方法getSite()
和getPrice().Component
參與者具體為具體組件和Decorator參與者抽象類建立接口:
IComponent.php
<?php abstract class IComponent { protected $site; abstract public function getSite(); abstract public function getPrice(); }
Decorator接口
這個(gè)例子中的裝飾器接口可能會(huì)讓你驚訝.這是一個(gè)抽象類,而且它還擴(kuò)展了另一個(gè)抽象類! 這個(gè)類的作用就是維護(hù)組件接口(IComponent)的一個(gè)引用, 這是通過擴(kuò)展IComponent完成的:
Decorator.php
<?php abstract class Decorator extends IComponent { /* 任務(wù)是維護(hù)Component的引用 繼承g(shù)etSite()和getPrice() 因?yàn)槿匀皇浅橄箢?所以不需要實(shí)現(xiàn)父類任何一個(gè)抽象方法 */ }
Decorator類的主要作用就是維護(hù)組件接口的一個(gè)引用.
在所有的裝飾器模式實(shí)現(xiàn)中, 你會(huì)發(fā)現(xiàn),具體組件和裝飾順都有相同的接口. 它們的實(shí)現(xiàn)可能不同, 另外除了基本接口的屬性和方法外, 組件和裝飾器可能還有額外的屬性和方法.
具體組件
這個(gè)例子中只有一個(gè)具體組件,它生成一個(gè)網(wǎng)站名, 另外生成一個(gè)基本網(wǎng)站報(bào)價(jià):
BasicSite.php
<?php class BasicSite extends IComponent { public function __construct() { $this->site = "Basic Site"; } public function getSite() { return $this->site; } public function getPrice() { return 1200; } }
兩個(gè)抽象方法都使用直接賦值來實(shí)現(xiàn), 不過靈活性并不體現(xiàn)在如何改變?cè)O(shè)置的值.實(shí)際上, 要通過增加裝飾器值來改變"Basic Site"值.
具體裝飾器
這個(gè)例子中的具體裝飾器與具體組件有相同的接口.實(shí)際上, 它們是從Decorator抽象類(而不是IComponent類)繼承了這個(gè)接口. 不過,要記住, Decorator所做的就是繼承IComponent接口.
Maintenance.php
<?php class Maintenance extends Decorator { public function __construct(IComponent $siteNow) { $this->site = $siteNow; } public function getSite() { $format = "<br /> Maintenance"; return $this->site->getSite() . $format; } public function getPrice() { return 950 + $this->site->getPrice(); } }
這個(gè)裝飾器Maintenance在改變了site的值, 還有包裝的具體組件價(jià)格上還會(huì)增加它自己 的價(jià)格. 另個(gè)兩個(gè)具體裝飾器與Maintenance裝飾器也類似
Video.php
<?php class Video extends Decorator { public function __construct(IComponent $siteNow) { $this->site = $siteNow; } public function getSite() { $format = "<br /> Video"; return $this->site->getSite() . $format; } public function getPrice() { return 350 + $this->site->getPrice(); } }
DataBase.php
<?php class DataBase extends Decorator { public function __construct(IComponent $siteNow) { $this->site = $siteNow; } public function getSite() { $format = "<br /> DataBase"; return $this->site->getSite() . $format; } public function getPrice() { return 800 + $this->site->getPrice(); } }
測(cè)試這個(gè)應(yīng)用時(shí),可以看到,在基本的價(jià)格之上還會(huì)增加各個(gè)裝飾器的價(jià)格.另外還能指定裝飾器名的格式, 增加了兩個(gè)空格,使之縮進(jìn)
裝飾器實(shí)現(xiàn)中最重要的元素之五就是構(gòu)造函數(shù), 要為構(gòu)造函數(shù)提供一個(gè)組件類型. 由于這里只有一個(gè)具體組件, 所有裝飾器的實(shí)例化都會(huì)使用這個(gè)組件. 使用多個(gè)組件時(shí), 裝飾器可以包裝應(yīng)用中的一部分或全部組件, 也可以不包裝任何組件.
客戶
Client類并不是這個(gè)設(shè)計(jì)模式的一部分, 但是正確使用Client類至關(guān)重要.每個(gè)裝飾器在實(shí)例化時(shí)"包裝"組件, 不過, 首先必須創(chuàng)建一個(gè)要包裝的對(duì)象, 這里是BasicSite類實(shí)例
Client.php
<?php function __autoload($class_name) { include $class_name . '.php'; } class Client { private $basicSite; public function __construct() { $this->basicSite = new BasicSite(); $this->basicSite = $this->WrapComponent($this->basicSite); $siteShow = $this->basicSite->getSite(); $format = "<br /> <strong>Total= $"; $price = $this->basicSite->getPrice(); echo $siteShow . $format . $price . "</strong>"; } private function WrapComponent(IComponent $component) { $component = new Maintenance($component); $component = new Video($component); $component = new DataBase($component); return $component; } } $worker = new Client();
wrapComponent()
方法檢查傳入的BasicSite實(shí)例, 以確保參數(shù)有正確的數(shù)據(jù)類型(IComponent), 然后分別實(shí)例化3個(gè)裝飾器, 對(duì)該實(shí)例對(duì)象進(jìn)行裝飾.
Basic Site
Maintenance
Video
DataBase
Total= $3300
適配器和裝飾器模式都有另外一個(gè)名字"包裝器"(wrapper)".
適配器可以"包裝"一個(gè)對(duì)象, 創(chuàng)建一個(gè)與Adaptee兼容的接口, 而無須對(duì)它做任何修改.
裝飾器也可以"包裝"一個(gè)組件對(duì)象, 這樣就能為這個(gè)已胡的組件增加職責(zé), 而無須對(duì)它做任何修改.
下面的代碼展示了Client如何將組件對(duì)象($component)包裝在裝飾器(Maintence)中:
$component = new Maintenance($component);
類似于"接口", 在計(jì)算機(jī)編程中用到"包裝器"時(shí), 不同的上下文會(huì)有不同的用法和含義. 一般來講, 在設(shè)計(jì)模式中使用"包裝器"是為了處理接口的不兼容, 或者希望為組件增加功能,包裝器就表示用來減少不兼容性的策略.
更多關(guān)于PHP相關(guān)內(nèi)容感興趣的讀者可查看本站專題:《php面向?qū)ο蟪绦蛟O(shè)計(jì)入門教程》、《PHP基本語法入門教程》、《PHP數(shù)組(Array)操作技巧大全》、《php字符串(string)用法總結(jié)》、《php+mysql數(shù)據(jù)庫操作入門教程》及《php常見數(shù)據(jù)庫操作技巧匯總》
希望本文所述對(duì)大家PHP程序設(shè)計(jì)有所幫助。
相關(guān)文章
PHP5中虛函數(shù)的實(shí)現(xiàn)方法分享
學(xué)過C++的人都應(yīng)該知道C++中有個(gè)虛函數(shù)的概念。而在php5中如何實(shí)現(xiàn)這個(gè)虛函數(shù)呢?2011-04-04通過緩存數(shù)據(jù)庫結(jié)果提高PHP性能的原理介紹
眾所周知,緩存數(shù)據(jù)庫查詢的結(jié)果可以顯著縮短腳本執(zhí)行時(shí)間,并最大限度地減少數(shù)據(jù)庫服務(wù)器上的負(fù)載。如果要處理的數(shù)據(jù)基本上是靜態(tài)的,則該技術(shù)將非常有效。這是因?yàn)閷?duì)遠(yuǎn)程數(shù)據(jù)庫的許多數(shù)據(jù)請(qǐng)求最終可以從本地緩存得到滿足,從而不必連接到數(shù)據(jù)庫、執(zhí)行查詢以及獲取結(jié)果2012-09-09php 下載保存文件保存到本地的兩種實(shí)現(xiàn)方法
以下是對(duì)php下載保存文件保存到本地的兩種實(shí)現(xiàn)方法進(jìn)行了介紹,需要的朋友可以過來參考下2013-08-08你應(yīng)該知道PHP浮點(diǎn)數(shù)知識(shí)
這篇文章主要介紹了你應(yīng)該知道PHP浮點(diǎn)數(shù)知識(shí),本文講解了PHP浮點(diǎn)數(shù)、PHP數(shù)字的臨界值,精度損失等問題,需要的朋友可以參考下2015-05-05根據(jù)key刪除數(shù)組中指定的元素實(shí)現(xiàn)方法
下面小編就為大家?guī)硪黄鶕?jù)key刪除數(shù)組中指定的元素實(shí)現(xiàn)方法。小編覺得挺不錯(cuò)的,現(xiàn)在就分享給大家,也給大家做個(gè)參考。一起跟隨小編過來看看吧2017-03-03PHP數(shù)組操作實(shí)例分析【添加,刪除,計(jì)算,反轉(zhuǎn),排序,查找等】
這篇文章主要介紹了PHP數(shù)組操作,結(jié)合實(shí)例形式分析php針對(duì)數(shù)組的添加,刪除,計(jì)算,反轉(zhuǎn),排序,查找等操作實(shí)現(xiàn)技巧,需要的朋友可以參考下2016-12-12解析PHP計(jì)算頁面執(zhí)行時(shí)間的實(shí)現(xiàn)代碼
本篇文章是對(duì)PHP計(jì)算頁面執(zhí)行時(shí)間的實(shí)現(xiàn)代碼進(jìn)行了詳細(xì)的分析介紹,需要的朋友參考下2013-06-06php數(shù)組函數(shù)序列之a(chǎn)rray_flip() 將數(shù)組鍵名與值對(duì)調(diào)
array_flip() 函數(shù)將使數(shù)組的鍵名與其相應(yīng)值調(diào)換,即鍵名變成了值,而值變成了鍵名2011-11-11