設計模式中,模闆模式面向的是方法級别的流程。(不過好像世界上大部分問題,都可以抽象點、抽象點吧,最後抽象到一個方法裡面吧。) 1. 一個方法,可以用來描述一個流程,這個流程涉及多個環節,不同環節可以用不同方法實作來做。 2. 同時,這些小環節也是 【方法所在類】 對外的接口,也可以被外部調用。 3. 由1、2可得,那些抽象的方法接口,我們就在【繼承】的子類中實作吧。當然是【繼承】,因為模闆方法的每一個抽象接口都必須被實作。 java web程式員對這個應該也很熟悉。我們的spring MVC架構中, dispatcher Servlet處理http請求不是一整套流程嗎。這些流程中,用到了多個子產品,然後每個子產品,我們不是都可以自己實作再注入嗎。呵呵,就這個意思。
1.概述
在面向對象開發過程中,通常我們會遇到這樣的一個問題:我們知道一個算法所需的關鍵步驟,并确定了這些步驟的執行順序。但是某些步驟的具體實作是未知的,或者說某些步驟的實作與具體的環境相關。 例子1:銀行業務辦理流程 在銀行辦理業務時,一般都包含幾個基本固定步驟: 取号排隊->辦理具體業務->對銀行從業人員進行評分。 取号取号排隊和對銀行從業人員進行評分業務邏輯是一樣的。但是辦理具體業務是個不相同的,具體業務可能取款、存款或者轉賬。
2.問題
如何保證架構邏輯的正常執行,而不被子類破壞 ?
3.解決方案
模闆方法:定義一個操作中的算法的骨架,而将一些步驟延遲到子類中。 T模闆方法使得子類可以不改變一個算法的結構即可重定義該算法的某些特定步驟。(Template Method Pattern:Definethe skeleton of an algorithm in an operation,deferring some steps tosubclasses.Template Methodletssubclasses redefine certain steps of an algorithmwithoutchanging the algorithm's structure. ) 1)模闆方法模式是基于繼承的代碼複用基本技術,模闆方法模式的結構和用法也是面向對象設計的核心之一。在模闆方法模式中,可以将相同的代碼放在父類中,而将不同的方法實作放在不同的子類中。 2)在模闆方法模式中,我們需要準備一個抽象類,将部分邏輯以具體方法以及具體構造函數的形式實作,然後聲明一些抽象方法來讓子類實作剩餘的邏輯。不同的子類可以以不同的方式實作這些抽象方法,進而對剩餘的邏輯有不同的實作,這就是模闆方法模式的用意。模闆方法模式展現了面向對象的諸多重要思想,是一種使用頻率較高的模式。
4.适用性
模闆方法應用于下列情況: • 1) 一次性實作一個算法的不變的部分,并将可變的行為留給子類來實作。 • 2)各子類中公共的行為應被提取出來并集中到一個公共父類中以避免代碼重複。首先識别現有代碼中的不同之處,并且将不同之處分離為新的操作。最後,用一個調用這些新的操作的模闆方法來替換這些不同的代碼。 • 3)控制子類擴充。模闆方法隻在特定點調用“ hook”操作 ,這樣就隻允許在這些點進行擴充。
5.結構
6.模式的組成
抽象類(AbstractClass): 定義抽象的原語操作(primitive operation) ,具體的子類将重定義它們以實作一個算法, 實作一個模闆方法,定義一個算法的骨架。該模闆方法不僅調用原語操作,也調用定義 具體子類 (ConcreteClass): 實作原語操作以完成算法中與特定子類相關的步驟。
7.效果
模闆方法模式的優點: 1)模闆方法模式在一個類中形式化地定義算法,而由它的子類實作細節的處理。 2)模闆方法是一種代碼複用的基本技術。它們在類庫中尤為重要,它們提取了類庫中的公共行為。 3)模闆方法模式導緻一種反向的控制結構,這種結構有時被稱為“好萊塢法則” ,即“别找我們,,我們找你”通過一個父類調用其子類的操作(而不是相反的子類調用父類),通過對子類的擴充增加新的行為,符合“開閉原則” 模闆方法模式的缺點: 每個不同的實作都需要定義一個子類,這會導緻類的個數增加,系統更加龐大,設計也更加抽象,但是更加符合“單一職責原則”,使得類的内聚性得以提高。
8.實作
我們使用銀行業務實作,添加了hook方法,客戶自己覺得評價服務。 <?php /** * 模闆方法模式: * * @author guisu */ * 抽象類 abstract class AbstractBank { private $_number ; /** *模闆方法 * 因為子類不能覆寫一個被定義為final的方法。進而保證了子類的邏輯永遠由父類所控制。 * */ public final function templateMethodProcess() { $this->takeNumber(); $this->transact(); if($this->isEvaluateHook()) { $this->evaluateHook(); } } * 基本方法—具體方法 * 取号 private function takeNumber() return ++$this->_number; * //基本方法—抽象方法 protected abstract function transact(); * 基本方法—鈎子方法 protected function evaluateHook() { echo ' evaluateHook<br/>'; } protected function isEvaluateHook() { return true; } * 具體子類:存款 class ConcreteDeposit extends AbstractBank public function transact() { //實作代碼 echo 'Deposit', '<br>'; * 具體子類:取款 class ConcreteWithdraw extends AbstractBank public function transact() { echo 'Withdraw', '<br>'; * 具體子類:轉賬 class ConcreteTrancfer extends AbstractBank echo 'Trancfer', '<br>'; $c = new ConcreteTrancfer(); $c->templateMethodProcess();
9.與其他相關模式
1)政策模式:模闆方法使用繼承來改變算法的一部分。 Strategy使用委托來改變整個算法。模闆方法模式與政策模式的作用十分類似,有時可以用政策模式替代模闆方法模式。模闆方法模式通過繼承來實作代碼複用,而政策模式使用委托,把不确定的行為集中到一個接口中,并在主類委托這個接口。委托比繼承具有更大的靈活性。
10.模式的擴充
1)模闆方法模式與控制反轉(好萊塢原則)在模闆方法模式中,子類不顯式調用父類的方法,而是通過覆寫父類的方法來實作某些具體的業務邏輯,父類控制對子類的調用,這種機制被稱為好萊塢原則(Hollywood Principle),好萊塢原則的定義為:“不要給我們打電話,我們會給你打電話(Don‘t call us, we’ll call you)”。在好萊塢,把履歷遞交給演藝公司後就隻有回家等待。由演藝公司對整個娛樂項的完全控制,演員隻能被動式的接受公司的差使,在需要的環節中,完成自己的演出。模闆方法模式充分的展現了“好萊塢”原則。由父類完全控制着子類的邏輯,子類不需要調用父類,而通過父類來調用子類,子類可以實作父類的可變部份,卻繼承父類的邏輯,不能改變業務邏輯。 2)模闆方法模式符合開閉原則 模闆方法模式意圖是由抽象父類控制頂級邏輯,并把基本操作的實作推遲到子類去實作,這是通過繼承的手段來達到對象的複用,同時也遵守了開閉原則。 父類通過頂級邏輯,它通過定義并提供一個具體方法來實作,我們也稱之為模闆方法。通常這個模闆方法才是外部對象最關心的方法。在上面的銀行業務處理例子中,templateMethodProcess這個方法才是外部對象最關心的方法。是以它必須是public的,才能被外部對象所調用。 子類需要繼承父類去擴充父類的基本方法,但是它也可以覆寫父類的方法。如果子類去覆寫了父類的模闆方法,進而改變了父類控制的頂級邏輯,這違反了“開閉原則”。我們在使用模闆方法模式時,應該總是保證子類有正确的邏輯。是以模闆方法應該定義為final的。是以AbstractClass類的模闆方法templateMethodProcess方法應該定義為final。 模闆方法模式中,抽象類的模闆方法應該聲明為final的。因為子類不能覆寫一個被定義為final的方法。進而保證了子類的邏輯永遠由父類所控制。
3)模闆方法模式與對象的封裝性
面向對象的三大特性:繼承,封裝,多态。 對象有内部狀态和外部的行為。封裝是為了資訊隐藏,通過封裝來維護對象内部資料的完整性。使得外部對象不能夠直接通路一個對象的内部狀态,而必須通過恰當的方法才能通路。 對象屬性和方法賦予指定的修改符(public、protected、private)來達到封裝的目的,使得資料不被外部對象惡意的通路及方法不被錯誤調用導造成破壞對象的封裝性。 降低方法的通路級别,也就是最大化的降低方法的可見度是一種很重要的封裝手段。最大化降低方法的可見度除了可以達到資訊隐藏外,還能有效的降低類之間的耦合度,降低一個類的複雜度。還可以減少開發人員發生的的錯誤調用。 一個類應該隻公開外部需要調用的方法。而所有為public方法服務的方法都應該聲明為protected或private。如是一個方法不是需要對外公開的,但是它需要被子類進行擴充的或調用。那麼把它定義為protected.否則應該為private。 顯而易見,模闆方法模式中的聲明為abstract的基本操作都是需要迫使子類去實作的,它們僅僅是為模闆方法服務的。它們不應該被抽象類(AbstractClass)所公開,是以它們應該protected。 是以模闆方法模式中,迫使子類實作的抽象方法應該聲明為protected abstract。
4)模闆方法與勾子方法(hookMethod) 模闆方法模式的抽象類定義方法: 模闆方法:一個模闆方法是定義在抽象類中的、把基本操作方法組合在一起形成一個總算法或一個總行為的方法。 基本方法:基本方法是實作算法各個步驟的方法,是模闆方法的組成部分。基本方法如下: •抽象方法(Abstract Method) •具體方法(Concrete Method) •鈎子方法(Hook Method):“挂鈎”方法和空方法, hook方法在抽象類中的實作為空,是留給子類做一些可選的操作。如果某個子類需要一些特殊額外的操作,則可以實作hook方法,當然也可以完全不用理會,因為hook在抽象類中隻是空方法而已。 1)鈎子方法的引入使得子類可以控制父類的行為。 2)最簡單的鈎子方法就是空方法,也可以在鈎子方法中定義一個預設的實作,如果子類不覆寫鈎子方法,則執行父類的預設實作代碼。 3)比較複雜一點的鈎子方法可以對其他方法進行限制,這種鈎子方法通常傳回一個boolean類型,即傳回true或false,用來判斷是否執行某一個基本方法。由子類來決定是否調用hook方法。
11.總結與分析
1)模闆方法模式是一種類的行為型模式,在它的結構圖中隻有類之間的繼承關系,沒有對象關聯關系。 2)闆方法模式是基于繼承的代碼複用基本技術,模闆方法模式的結構和用法也是面向對象設計的核心之一。在模闆方法模式中,可以将相同的代碼放在父類中,而将不同的方法實作放在不同的子類中。 3)在模闆方法模式中,我們需要準備一個抽象類,将部分邏輯以具體方法以及具體構造函數的形式實作,然後聲明一些抽象方法來讓子類實作剩餘的邏輯。不同的子類可以以不同的方式實作這些抽象方法,進而對剩餘的邏輯有不同的實作,這就是模闆方法模式的用意。模闆方法模式展現了面向對象的諸多重要思想,是一種使用頻率較高的模式。
本文轉自二郎三郎部落格園部落格,原文連結:http://www.cnblogs.com/haore147/p/5710462.html,如需轉載請自行聯系原作者