摘要:重構(gòu)時,模板方法模式是一個經(jīng)常使用的模式,把相同的代碼抽取到父類中,然后通過鉤子函數(shù)詳見后面的擴展示例約束其行為。
定義
Define the skeleton of an algorithm in an operation,deferring some steps to subclasses.Template
Method lets subclasses redefine certain steps of an algorithm without changing the algorithm"s
structure.定義一個操作中的算法的框架,而將一些步驟延遲到子類中。使得子類可以不改
變一個算法的結(jié)構(gòu)即可重定義該算法的某些特定步驟。
模板方法模式非常簡單,主要是用了Java的繼承機制,話不多說,直接上代碼
實現(xiàn) 抽象模板類public abstract class AbstractClass { /** * 基本方法 */ protected abstract void doSomething(); /** * 基本方法,可以有默認實現(xiàn) */ protected void doAnything() { System.out.println("AbstractClass doAnything()"); } /** * 模板方法,為了防止惡意的操作,一般模板方法都加上final關(guān)鍵字,不允許被覆寫 */ public final void templateMethod(){ doSomething(); doAnything(); } }具體模板類
public class ConcreteClassA extends AbstractClass { @Override protected void doSomething() { System.out.println("ConcreteClassA doSomething()"); } @Override protected void doAnything() { System.out.println("ConcreteClassA doAnything()->我不想使用父類的默認實現(xiàn),我要覆蓋它"); } }
public class ConcreteClassB extends AbstractClass { @Override protected void doSomething() { System.out.println("ConcreteClassB doSomething()"); } // 使用父類doAnything()的默認實現(xiàn) }客戶端代碼
public class Client { public static void main(String[] args) { AbstractClass a = new ConcreteClassA(); a.templateMethod(); AbstractClass b = new ConcreteClassB(); b.templateMethod(); } }優(yōu)點
封裝不變部分,擴展可變部分
提取公共部分代碼,便于維護
行為由父類控制,子類實現(xiàn)
缺點子類影響父類
按照我們的設(shè)計習慣,抽象類負責聲明最抽象、最一般的事物屬性和方法,實現(xiàn)類完成
具體的事物屬性和方法。但是模板方法模式卻顛倒了,抽象類定義了部分抽象方法,由子類
實現(xiàn),子類執(zhí)行的結(jié)果影響了父類的結(jié)果,也就是子類對父類產(chǎn)生了影響,這在復雜的項目
中,會帶來代碼閱讀的難度,而且也會讓新手產(chǎn)生不適感。
模板方法使用繼承方式復用代碼,如果要在基本算法里面增加一個步驟,而該步驟是抽象的話,每個子類都要修改代碼,實現(xiàn)這個步驟。
使用場景多個子類有公有的方法,并且邏輯基本相同時。
重要、復雜的算法,可以把核心算法設(shè)計為模板方法,周邊的相關(guān)細節(jié)功能則由各個
子類實現(xiàn)。
重構(gòu)時,模板方法模式是一個經(jīng)常使用的模式,把相同的代碼抽取到父類中,然后通
過鉤子函數(shù)(詳見后面的擴展示例)約束其行為。
擴展模板方法模式的擴展,主要就是增加了鉤子方法(Hook Method),那么什么是“鉤子方法”呢?
在抽象模板類中,可以定義一個方法,并允許子類視情況覆蓋它來改變基本方法的執(zhí)行過程(比如決定某些步驟是否需要執(zhí)行)
鉤子方法的作用
讓子類實現(xiàn)算法中的可選部分,算法中的某些步驟是可選的,子類可以做出決定是否需要這些步驟
如果鉤子對于子類的實現(xiàn)不重要時,子類可以對鉤子置之不理
下面是增加鉤子方法后的模板方法模式通用代碼:
抽象模板類public abstract class AbstractClass { /** * 基本方法 */ protected abstract void doSomething(); /** * 基本方法 */ protected void doAnything() { System.out.println("AbstractClass doAnything()"); } /** * 依賴于鉤子方法的基本方法 */ protected abstract void dependOnHook(); /** * 模板方法,為了防止惡意的操作,一般模板方法都加上final關(guān)鍵字,不允許被覆寫 */ public final void templateMethod(){ doSomething(); doAnything(); if (hook()){ dependOnHook(); } } /** * 鉤子方法:空實現(xiàn)或默認實現(xiàn),子類可以覆寫;由子類的一個方法返回值決定公共部分的執(zhí)行結(jié)果 * @return */ protected boolean hook(){ System.out.println("AbstractClass hook()"); return true; } }具體模板類
public class ConcreteClassA extends AbstractClass { @Override protected void doSomething() { System.out.println("ConcreteClassA doSomething()"); } @Override protected void doAnything() { System.out.println("ConcreteClassA doAnything()->我不想使用父類的默認實現(xiàn),我要覆蓋它"); } @Override protected void dependOnHook() { System.out.println("ConcreteClassA dependOnHook()"); } // 沒有覆寫鉤子方法,使用默認實現(xiàn),dependOnHook()將會被調(diào)用 }
public class ConcreteClassB extends AbstractClass { @Override protected void doSomething() { System.out.println("ConcreteClassB doSomething()"); } // 使用父類doAnything()的默認實現(xiàn) @Override protected void dependOnHook() { System.out.println("ConcreteClassB dependOnHook()"); } /** * 覆寫鉤子方法,改變默認實現(xiàn),改變公共部分(模板方法)的行為,dependOnHook()不會被調(diào)用 * @return */ @Override protected boolean hook(){ System.out.println("ConcreteClassB hook()"); return false; } }
源碼地址:https://gitee.com/tianranll/java-design-patterns.git
參考文獻《設(shè)計模式之禪》
文章版權(quán)歸作者所有,未經(jīng)允許請勿轉(zhuǎn)載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。
轉(zhuǎn)載請注明本文地址:http://systransis.cn/yun/77893.html
摘要:模板方法模式定義定義抽象類并且聲明一些抽象基本方法供子類實現(xiàn)不同邏輯,同時在抽象類中定義具體方法把抽象基本方法封裝起來,這就是模板方法模式。 近日,ofo小黃車宣布入駐法國巴黎,正式進入全球第20個國家,共享單車已然改變了我們的出行方式。就拿我自己來說,每當下班出地鐵的第一件事,以光速鎖定一輛共享單車,百米沖刺的速度搶在別人之前占領(lǐng)它。 而大家都是重復著同樣的動作,拿出手機開鎖、騎車、...
摘要:模板方法模式定義了一個算法的步驟,并允許子類別為一個或多個步驟提供其實踐方式。在軟件工程中,它是一種軟件設(shè)計模式,和模板沒有關(guān)連。模板方法充分運用了多態(tài)與繼承。去建設(shè)銀行支付去招商銀行支付實現(xiàn)模板方法的細節(jié),我們來看使用邏輯。 Photo by Tomá? Malík on Unsplash 什么是模板方法模式?摘錄 wiki 的介紹。 模板方法模式定義了一個算法的步驟,并允許子類別為...
摘要:目錄建造者模式應(yīng)用。其實不用也可以,因為不是很復雜,只是為了復習一下所學過的設(shè)計模式知識目錄工廠模式應(yīng)用。 為了提高開發(fā)效率,通常會想辦法把一些模式固定的重復性的勞動抽取出來,以后再使用的時候,拿來主義就可以了。這樣既可以提高開發(fā)效率,又降低了出錯的風險。 這一思想在我們的日常工作中可以說隨處可見,我們完成一項復雜的工程,并不需要面面俱到什么都自己寫,我們完全可以利用第三方的jar包讓...
摘要:讓一切變得更簡單抽象化深入封裝算法塊,這便是設(shè)計模式當中的一種模式模板方法模式。定義模板方法模式在一個方法中定義一個算法的框架,而將一些步驟延遲到子類中。 讓一切變得更簡單抽象化 深入封裝算法塊,這便是設(shè)計模式當中的一種模式:模板方法模式。 我們先來看看下面兩個茶和咖啡配方showImg(https://segmentfault.com/img/bVV4kS?w=1248&h=...
閱讀 3445·2021-11-08 13:20
閱讀 3373·2021-09-30 09:48
閱讀 2574·2021-09-29 09:41
閱讀 596·2021-09-22 15:04
閱讀 2485·2021-08-23 09:44
閱讀 3686·2020-12-03 17:26
閱讀 1017·2019-08-30 14:10
閱讀 1573·2019-08-29 18:34