本文實(shí)例講述了PHP設(shè)計(jì)模式之模板方法模式定義與用法。分享給大家供大家參考,具體如下:
什么是模板方法模式
模板方法(Template Method)設(shè)計(jì)模式中使用了一個(gè)類方法templateMethod(), 該方法是抽象類中的一個(gè)具體方法, 這個(gè)方法的作用是對(duì)抽象方法序列排序,具體實(shí)現(xiàn)留給具體類來(lái)完成.關(guān)鍵在于模板方法模式定義了操作中算法的"骨架",而由具體類來(lái)實(shí)現(xiàn).
什么時(shí)候使用模板方法
如果已經(jīng)明確算法中的一些步驟, 不過(guò)這些步驟可以采用多種不同的方法實(shí)現(xiàn), 就可以使用模板方法調(diào)試.如果算法中的步驟不變, 可以把這些步驟留給子類具體實(shí)現(xiàn).在這種情況下, 可以使用模板方法設(shè)計(jì)模式來(lái)組織抽象類中的基本操作(函數(shù)/方法).然后由子類來(lái)實(shí)現(xiàn)應(yīng)用所需的這些操作.
還有一種用法稍微復(fù)雜一些, 可能需要把子類共同的行為放在一個(gè)類中, 以避免代碼重復(fù).
如果使用多個(gè)類來(lái)解決同一個(gè)大型問(wèn)題, 可能很快就會(huì)出現(xiàn)重復(fù)代碼.
還有一點(diǎn),可以使用模板方法模式控制子類擴(kuò)展,也就是所謂的"鉤子".
示例
在PHP編程中,可能經(jīng)常會(huì)遇到一個(gè)問(wèn)題: 要建立帶圖題的圖像. 這個(gè)算法相當(dāng)簡(jiǎn)單, 就是顯示圖像, 然后的圖像下面顯示文本.
由于模板設(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
?php
abstract 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
?php
include_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
?php
function __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模板方法, 這是從父類繼承的具體操作, 父類通過(guò)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è)方法作為模板的一部分,不過(guò)不一定會(huì)用到這個(gè)方法, 換句話說(shuō), 它是方法的一部分,不過(guò)它包含一個(gè)鉤子, 可以處理例外情況. 子類可以為算法增加一個(gè)可選元素, 這樣一來(lái), 盡管仍按模板方法建立的順序執(zhí)行, 但有可能并不完成模板方法期望的動(dòng)作. 對(duì)于這種可選的情況, 鉤子就是解決這個(gè)問(wèn)題最理想的工具.
示例
去網(wǎng)購(gòu)商品, 登場(chǎng)8折, 如果總商品費(fèi)用超過(guò)200元, 就免去12.95元錢運(yùn)費(fèi).
建立鉤子
在模板方法中建立鉤子方法很有意思, 盡管子類可以改變鉤子的行為, 但仍然要遵循模板中定義的順序
IHook.php
?php
abstract 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è)變量用來(lái)確定顧客是否免收運(yùn)費(fèi).
實(shí)現(xiàn)鉤子
一旦抽象類中建立了這些抽象方法, 并指定了它們執(zhí)行的順序, 子類將實(shí)現(xiàn)所有這3個(gè)方法:
Concrete.php
?php
class 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)., 不過(guò), addShippingHook的實(shí)現(xiàn)有所不同, 其中有一個(gè)條件來(lái)確定是否增加運(yùn)費(fèi). 這就是鉤子.
客戶Client
Client.php
?php
function __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演示了分別購(gòu)買100塊錢和200塊錢的商品最后的費(fèi)用,運(yùn)行結(jié)果如下
您需要支付: 92.95元
您需要支付: 160元
更多關(guān)于PHP相關(guān)內(nèi)容感興趣的讀者可查看本站專題:《php面向?qū)ο蟪绦蛟O(shè)計(jì)入門教程》、《PHP基本語(yǔ)法入門教程》、《PHP數(shù)組(Array)操作技巧大全》、《php字符串(string)用法總結(jié)》、《php+mysql數(shù)據(jù)庫(kù)操作入門教程》及《php常見(jiàn)數(shù)據(jù)庫(kù)操作技巧匯總》
希望本文所述對(duì)大家PHP程序設(shè)計(jì)有所幫助。
您可能感興趣的文章:- 學(xué)習(xí)php設(shè)計(jì)模式 php實(shí)現(xiàn)模板方法模式
- PHP設(shè)計(jì)模式之模板模式定義與用法詳解
- PHP設(shè)計(jì)模式之觀察者模式(Observer)詳細(xì)介紹和代碼實(shí)例
- php設(shè)計(jì)模式 Template (模板模式)
- php設(shè)計(jì)模式 DAO(數(shù)據(jù)訪問(wèn)對(duì)象模式)
- php設(shè)計(jì)模式 Proxy (代理模式)
- php單態(tài)設(shè)計(jì)模式(單例模式)實(shí)例
- php設(shè)計(jì)模式 Singleton(單例模式)
- php設(shè)計(jì)模式 Facade(外觀模式)
- php設(shè)計(jì)模式 Strategy(策略模式)
- PHP設(shè)計(jì)模式之模板方法模式實(shí)例淺析