本文實(shí)例講述了PHP設(shè)計(jì)模式之模板方法模式定義與用法。分享給大家供大家參考,具體如下:
什么是模板方法模式
模板方法(Template Method)設(shè)計(jì)模式中使用了一個(gè)類方法templateMethod(), 該方法是抽象類中的一個(gè)具體方法, 這個(gè)方法的作用是對抽象方法序列排序,具體實(shí)現(xiàn)留給具體類來完成.關(guān)鍵在于模板方法模式定義了操作中算法的"骨架",而由具體類來實(shí)現(xiàn).
什么時(shí)候使用模板方法
如果已經(jīng)明確算法中的一些步驟, 不過這些步驟可以采用多種不同的方法實(shí)現(xiàn), 就可以使用模板方法調(diào)試.如果算法中的步驟不變, 可以把這些步驟留給子類具體實(shí)現(xiàn).在這種情況下, 可以使用模板方法設(shè)計(jì)模式來組織抽象類中的基本操作(函數(shù)/方法).然后由子類來實(shí)現(xiàn)應(yīng)用所需的這些操作.
還有一種用法稍微復(fù)雜一些, 可能需要把子類共同的行為放在一個(gè)類中, 以避免代碼重復(fù).
如果使用多個(gè)類來解決同一個(gè)大型問題, 可能很快就會(huì)出現(xiàn)重復(fù)代碼.
還有一點(diǎn),可以使用模板方法模式控制子類擴(kuò)展,也就是所謂的"鉤子".
示例
在PHP編程中,可能經(jīng)常會(huì)遇到一個(gè)問題: 要建立帶圖題的圖像. 這個(gè)算法相當(dāng)簡單, 就是顯示圖像, 然后的圖像下面顯示文本.
由于模板設(shè)計(jì)中只涉及兩個(gè)參與者, 所以這是最容易理解的模式之一, 同時(shí)也非常有用. 抽象建立templateMethod(),并由具體類實(shí)現(xiàn)這個(gè)方法.
抽象類
抽象類是這里的關(guān)鍵, 因?yàn)樗瑫r(shí)包含具體和抽象方法. 模板方法往往是具體方法, 其操作是抽象的
兩個(gè)抽象方法分別是addPicture和addTitile,這兩個(gè)操作都包含一個(gè)參數(shù), 分別表示圖像的URL信息和圖像標(biāo)題.
Template.php
<?phpabstract class Template{ protected $picture; protected $title; public function display($pictureNow, $titleNow) { $this->picture = $pictureNow; $this->title = $titleNow; $this->addPicture($this->picture); $this->addTitle($this->title); } abstract protected function addPicture($picture); abstract protected function addTitle($title);}具體類
Concrete.php
<?phpinclude_once('Template.php');class Concrete extends Template{ protected function addPicture($picture) { $this->picture = 'picture/' . $picture; echo "圖像路徑為:" . $this->picture . '<br />'; } protected function addTitle($title) { $this->title = $title; echo "<em>標(biāo)題: </em>" . $this->title . "<br />"; }}客戶
Client.php
<?phpfunction __autoload($class_name){ include $class_name . '.php';}class Client{ public function __construct() { $title = "chenqionghe is a handsome boy"; $concrete = new Concrete(); $concrete->display('chenqionghe.png', $title); }}$worker = new Client(); $concrete變量實(shí)例化了Concrete, 但是它調(diào)用了display模板方法, 這是從父類繼承的具體操作, 父類通過display()調(diào)用子類的操作.
運(yùn)行后輸出
圖像路徑為:picture/chenqionghe.png
標(biāo)題: chenqionghe is a handsome boy
可以看到,客戶只需要提供圖像地址和標(biāo)題
模板方法設(shè)計(jì)模式中的鉤子
有時(shí)模板方法函數(shù)可能有一個(gè)你不想要的步驟, 某些特定情況下你可能不希望執(zhí)行這個(gè)步驟, 這時(shí)候就可以用到模板方法的鉤子.
在模板方法設(shè)計(jì)模式中, 利用鉤子可以將一個(gè)方法作為模板的一部分,不過不一定會(huì)用到這個(gè)方法, 換句話說, 它是方法的一部分,不過它包含一個(gè)鉤子, 可以處理例外情況. 子類可以為算法增加一個(gè)可選元素, 這樣一來, 盡管仍按模板方法建立的順序執(zhí)行, 但有可能并不完成模板方法期望的動(dòng)作. 對于這種可選的情況, 鉤子就是解決這個(gè)問題最理想的工具.
示例
去網(wǎng)購商品, 登場8折, 如果總商品費(fèi)用超過200元, 就免去12.95元錢運(yùn)費(fèi).
建立鉤子
在模板方法中建立鉤子方法很有意思, 盡管子類可以改變鉤子的行為, 但仍然要遵循模板中定義的順序
IHook.php
<?phpabstract class IHook{ protected $hook; protected $fullCost; public function templateMethod($fullCost, $hook) { $this->fullCost = $fullCost; $this->hook = $hook; $this->addGoods(); $this->addShippingHook(); $this->displayCost(); } protected abstract function addGoods(); protected abstract function addShippingHook(); protected abstract function displayCost();} 這里有3個(gè)抽象方法: addGoods(), addShippingHook(),displayCost(), 抽象類IHook實(shí)現(xiàn)的templateMethod()中確定了它們的順序. 在這里, 鉤子方法放在中間, 實(shí)際上模板方法指定的順序中, 鉤子可以放在任意位置. 模板方法需要兩個(gè)參數(shù), 一個(gè)是總花費(fèi), 另外還需要一個(gè)變量用來確定顧客是否免收運(yùn)費(fèi).
實(shí)現(xiàn)鉤子
一旦抽象類中建立了這些抽象方法, 并指定了它們執(zhí)行的順序, 子類將實(shí)現(xiàn)所有這3個(gè)方法:
Concrete.php
<?phpclass Concrete extends IHook{ protected function addGoods() { $this->fullCost = $this->fullCost * 0.8; } protected function addShippingHook() { if(!$this->hook) { $this->fullCost += 12.95; } } protected function displayCost() { echo "您需要支付: " . $this->fullCost . '元<br />'; }}addGoods和displayCost都是標(biāo)準(zhǔn)方法, 只有一個(gè)實(shí)現(xiàn)., 不過, addShippingHook的實(shí)現(xiàn)有所不同, 其中有一個(gè)條件來確定是否增加運(yùn)費(fèi). 這就是鉤子.
客戶Client
Client.php
<?phpfunction __autoload($class_name){ include $class_name . '.php';}class Client{ private $totalCost; private $hook; public function __construct($goodsTotal) { $this->totalCost = $goodsTotal; $this->hook = $this->totalCost >= 200; $concrete = new Concrete(); $concrete->templateMethod($this->totalCost, $this->hook); }}$worker = new Client(100);$worker = new Client(200);該Client演示了分別購買100塊錢和200塊錢的商品最后的費(fèi)用,運(yùn)行結(jié)果如下
您需要支付: 92.95元
您需要支付: 160元
希望本文所述對大家PHP程序設(shè)計(jì)有所幫助。
新聞熱點(diǎn)
疑難解答
圖片精選