Java設(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的繼承機(jī)制澳窑,話不多說,直接上代碼

實(shí)現(xiàn)

抽象模板類

public abstract class AbstractClass {

    /**
     * 基本方法
     */
    protected abstract void doSomething();

    /**
     * 基本方法,可以有默認(rèn)實(shí)現(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()->我不想使用父類的默認(rèn)實(shí)現(xiàn)榜聂,我要覆蓋它");
    }
}
public class ConcreteClassB extends AbstractClass {
    @Override
    protected void doSomething() {
        System.out.println("ConcreteClassB doSomething()");
    }
    
    // 使用父類doAnything()的默認(rèn)實(shí)現(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)點(diǎn)

  • 封裝不變部分,擴(kuò)展可變部分
  • 提取公共部分代碼嗓蘑,便于維護(hù)
  • 行為由父類控制峻汉,子類實(shí)現(xiàn)

缺點(diǎn)

  • 子類影響父類

    按照我們的設(shè)計習(xí)慣,抽象類負(fù)責(zé)聲明最抽象脐往、最一般的事物屬性和方法休吠,實(shí)現(xiàn)類完成
    具體的事物屬性和方法。但是模板方法模式卻顛倒了业簿,抽象類定義了部分抽象方法瘤礁,由子類
    實(shí)現(xiàn),子類執(zhí)行的結(jié)果影響了父類的結(jié)果梅尤,也就是子類對父類產(chǎn)生了影響柜思,這在復(fù)雜的項目
    中岩调,會帶來代碼閱讀的難度,而且也會讓新手產(chǎn)生不適感赡盘。

  • 模板方法使用繼承方式復(fù)用代碼号枕,如果要在基本算法里面增加一個步驟,而該步驟是抽象的話陨享,每個子類都要修改代碼葱淳,實(shí)現(xiàn)這個步驟。

使用場景

  • 多個子類有公有的方法抛姑,并且邏輯基本相同時赞厕。
  • 重要、復(fù)雜的算法定硝,可以把核心算法設(shè)計為模板方法皿桑,周邊的相關(guān)細(xì)節(jié)功能則由各個
    子類實(shí)現(xiàn)。
  • 重構(gòu)時蔬啡,模板方法模式是一個經(jīng)常使用的模式诲侮,把相同的代碼抽取到父類中,然后通
    鉤子函數(shù)(詳見后面的擴(kuò)展示例)約束其行為箱蟆。

擴(kuò)展

模板方法模式的擴(kuò)展浆西,主要就是增加了鉤子方法(Hook Method),那么什么是“鉤子方法”呢顽腾?

在抽象模板類中近零,可以定義一個方法,并允許子類視情況覆蓋它來改變基本方法的執(zhí)行過程(比如決定某些步驟是否需要執(zhí)行)

鉤子方法的作用

  • 讓子類實(shí)現(xiàn)算法中的可選部分抄肖,算法中的某些步驟是可選的久信,子類可以做出決定是否需要這些步驟
  • 如果鉤子對于子類的實(shí)現(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();
        }

    }

    /**
     * 鉤子方法:空實(shí)現(xiàn)或默認(rèn)實(shí)現(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()->我不想使用父類的默認(rèn)實(shí)現(xiàn),我要覆蓋它");
    }

    @Override
    protected void dependOnHook() {
        System.out.println("ConcreteClassA dependOnHook()");
    }

    // 沒有覆寫鉤子方法夭咬,使用默認(rèn)實(shí)現(xiàn)啃炸,dependOnHook()將會被調(diào)用
}
public class ConcreteClassB extends AbstractClass {
    @Override
    protected void doSomething() {
        System.out.println("ConcreteClassB doSomething()");
    }

    // 使用父類doAnything()的默認(rèn)實(shí)現(xiàn)

    @Override
    protected void dependOnHook() {
        System.out.println("ConcreteClassB dependOnHook()");
    }

    /**
     * 覆寫鉤子方法,改變默認(rèn)實(shí)現(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

參考文獻(xiàn)《設(shè)計模式之禪》

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個濱河市,隨后出現(xiàn)的幾起案子裹虫,更是在濱河造成了極大的恐慌肿嘲,老刑警劉巖,帶你破解...
    沈念sama閱讀 218,941評論 6 508
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件筑公,死亡現(xiàn)場離奇詭異雳窟,居然都是意外死亡,警方通過查閱死者的電腦和手機(jī)匣屡,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 93,397評論 3 395
  • 文/潘曉璐 我一進(jìn)店門封救,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人耸采,你說我怎么就攤上這事兴泥」び” “怎么了虾宇?”我有些...
    開封第一講書人閱讀 165,345評論 0 356
  • 文/不壞的土叔 我叫張陵,是天一觀的道長如绸。 經(jīng)常有香客問我嘱朽,道長,這世上最難降的妖魔是什么怔接? 我笑而不...
    開封第一講書人閱讀 58,851評論 1 295
  • 正文 為了忘掉前任搪泳,我火速辦了婚禮,結(jié)果婚禮上扼脐,老公的妹妹穿的比我還像新娘岸军。我一直安慰自己,他們只是感情好瓦侮,可當(dāng)我...
    茶點(diǎn)故事閱讀 67,868評論 6 392
  • 文/花漫 我一把揭開白布棋电。 她就那樣靜靜地躺著驻谆,像睡著了一般。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上抖坪,一...
    開封第一講書人閱讀 51,688評論 1 305
  • 那天,我揣著相機(jī)與錄音蛛枚,去河邊找鬼术荤。 笑死,一個胖子當(dāng)著我的面吹牛斋泄,可吹牛的內(nèi)容都是我干的杯瞻。 我是一名探鬼主播,決...
    沈念sama閱讀 40,414評論 3 418
  • 文/蒼蘭香墨 我猛地睜開眼炫掐,長吁一口氣:“原來是場噩夢啊……” “哼又兵!你這毒婦竟也來了?” 一聲冷哼從身側(cè)響起,我...
    開封第一講書人閱讀 39,319評論 0 276
  • 序言:老撾萬榮一對情侶失蹤沛厨,失蹤者是張志新(化名)和其女友劉穎宙地,沒想到半個月后,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體逆皮,經(jīng)...
    沈念sama閱讀 45,775評論 1 315
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡宅粥,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 37,945評論 3 336
  • 正文 我和宋清朗相戀三年,在試婚紗的時候發(fā)現(xiàn)自己被綠了电谣。 大學(xué)時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片秽梅。...
    茶點(diǎn)故事閱讀 40,096評論 1 350
  • 序言:一個原本活蹦亂跳的男人離奇死亡,死狀恐怖剿牺,靈堂內(nèi)的尸體忽然破棺而出企垦,到底是詐尸還是另有隱情,我是刑警寧澤晒来,帶...
    沈念sama閱讀 35,789評論 5 346
  • 正文 年R本政府宣布钞诡,位于F島的核電站,受9級特大地震影響湃崩,放射性物質(zhì)發(fā)生泄漏荧降。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,437評論 3 331
  • 文/蒙蒙 一攒读、第九天 我趴在偏房一處隱蔽的房頂上張望朵诫。 院中可真熱鬧,春花似錦薄扁、人聲如沸剪返。這莊子的主人今日做“春日...
    開封第一講書人閱讀 31,993評論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽脱盲。三九已至,卻和暖如春震放,著一層夾襖步出監(jiān)牢的瞬間宾毒,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 33,107評論 1 271
  • 我被黑心中介騙來泰國打工殿遂, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留诈铛,地道東北人。 一個月前我還...
    沈念sama閱讀 48,308評論 3 372
  • 正文 我出身青樓墨礁,卻偏偏與公主長得像幢竹,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個殘疾皇子恩静,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 45,037評論 2 355

推薦閱讀更多精彩內(nèi)容

  • 定義 模板方法模式是類的行為模式焕毫。準(zhǔn)備一個抽象類蹲坷,將部分邏輯以具體方法以及具體構(gòu)造函數(shù)的形式實(shí)現(xiàn),然后聲明一些抽象...
    步積閱讀 884評論 0 0
  • 【學(xué)習(xí)難度:★★☆☆☆邑飒,使用頻率:★★★☆☆】直接出處:模板方法模式梳理和學(xué)習(xí):https://github.co...
    BruceOuyang閱讀 725評論 0 2
  • 介紹 模板方法模式是編程中經(jīng)常用得到的模式循签。它定義了一個操作中的算法骨架,將某些步驟延遲到子類中實(shí)現(xiàn)疙咸。這樣县匠,新的子...
    GaaraZ閱讀 299評論 0 0
  • 一、什么是模板方法模式 在一個方法中定義一個算法的骨架撒轮,而將一些步驟延續(xù)到子類中乞旦。模板方法使得子類在不改變算法結(jié)構(gòu)...
    只道初見閱讀 358評論 0 0
  • 平安健康應(yīng)首要。 體態(tài)無礙財自得题山, 心舒暢享樂自在兰粉。 無怨無欲勿貪苦, 香火供奉心坦誠顶瞳。 舉頭三尺神明賜玖姑, 護(hù)佑虔...
    白三緣閱讀 614評論 0 0