本文實(shí)例講述了PHP設(shè)計(jì)模式之模板方法模式定義與用法。分享給大家供大家參考,具體如下:
什么是模板方法模式
模板方法(Template Method)設(shè)計(jì)模式中使用了一個(gè)類方法templateMethod(), 該方法是抽象類中的一個(gè)具體方法, 這個(gè)方法的作用是對(duì)抽象方法序列排序,具體實(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
?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模板方法, 這是從父類繼承的具體操作, 父類通過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)作. 對(duì)于這種可選的情況, 鉤子就是解決這個(gè)問題最理想的工具.
示例
去網(wǎng)購商品, 登場8折, 如果總商品費(fèi)用超過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è)變量用來確定顧客是否免收運(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)., 不過, addShippingHook的實(shí)現(xiàn)有所不同, 其中有一個(gè)條件來確定是否增加運(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演示了分別購買100塊錢和200塊錢的商品最后的費(fèi)用,運(yùn)行結(jié)果如下
您需要支付: 92.95元
您需要支付: 160元
更多關(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ì)有所幫助。
您可能感興趣的文章:- 學(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ù)訪問對(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í)例淺析