23种设计模式-模板方法模式

  1. 辉煌工程-制造悍马

原书中作者说的公司是做模型生产的,举的例子是做车辆的模型,先不考虑扩展性,设计类图10-1:


10-1

这个设计其实就基本上是最基础的,先定义出车模型的接口,然后各种模型实现该接口就好了,代码如下:

public abstract class HummerModel {
    //能发动
    public abstract void start();
    //停
    public abstract void stop();
    //喇叭
    public abstract void alarm();
    //引擎响
    public abstract void engineBoom();
    //跑
    public abstract void run();
}
public class HummerH1Model extends HummerModel {
    @Override
    public void start() {
        System.out.println("悍马H1发动。。。");
    }

    @Override
    public void stop() {
        System.out.println("悍马H1停车。。。");
    }

    @Override
    public void alarm() {
        System.out.println("悍马H1鸣笛。。。");
    }

    @Override
    public void engineBoom() {
        System.out.println("悍马H1引擎声音。。。");
    }

    @Override
    public void run() {
        //先发动汽车
        this.start();
        //引擎开始轰鸣
        this.engineBoom();
        //按喇叭
        this.alarm();
        //停车
        this.stop();
    }
}

public class HummerH2Model extends HummerModel {
    @Override
    public void start() {
        System.out.println("悍马H2发动。。。");
    }

    @Override
    public void stop() {
        System.out.println("悍马H2停车。。。");
    }

    @Override
    public void alarm() {
        System.out.println("悍马H2鸣笛。。。");
    }

    @Override
    public void engineBoom() {
        System.out.println("悍马H2引擎声音。。。");
    }

    @Override
    public void run() {
        //先发动汽车
        this.start();
        //引擎开始轰鸣
        this.engineBoom();
        //按喇叭
        this.alarm();
        //停车
        this.stop();
    }
}

这里很明显的感觉就是,两个子类的run方法都是完全相同的,那么这个run方法的实现应该出现在抽象类中,不应该在实现类中,抽象是所有一类的共性封装。

注意 在软件开发过程中,如果一段代码复制过两次,就需要对设计产生怀疑,架构师要明确的说明为什么相同的逻辑要出现两次或更多次

修改类图如下:


10-2

代码修改如下:

public abstract class HummerModel {
    //能发动
    public abstract void start();
    //停
    public abstract void stop();
    //喇叭
    public abstract void alarm();
    //引擎响
    public abstract void engineBoom();
    //跑
    public void run(){
        //先发动汽车
        this.start();
        //引擎开始轰鸣
        this.engineBoom();
        //按喇叭
        this.alarm();
        //停车
        this.stop();
    }
}
public class HummerH1Model extends HummerModel {
    @Override
    public void start() {
        System.out.println("悍马H1发动。。。");
    }

    @Override
    public void stop() {
        System.out.println("悍马H1停车。。。");
    }

    @Override
    public void alarm() {
        System.out.println("悍马H1鸣笛。。。");
    }

    @Override
    public void engineBoom() {
        System.out.println("悍马H1引擎声音。。。");
    }
}
public class HummerH2Model extends HummerModel {
    @Override
    public void start() {
        System.out.println("悍马H2发动。。。");
    }

    @Override
    public void stop() {
        System.out.println("悍马H2停车。。。");
    }

    @Override
    public void alarm() {
        System.out.println("悍马H2鸣笛。。。");
    }

    @Override
    public void engineBoom() {
        System.out.println("悍马H2引擎声音。。。");
    }
}

这就是简单的模板方法模式,将确定的执行顺序放在抽象类中,实现类只需要实现具体实现步骤中的方法就好了。

  1. 模板方法模式的定义

模板方法模式(Template Method Pattern) 是如此简单,其定义如下:
Define the skeleton of algorthm in an operation,deferring some steps to subclasses.Template Method lets subclasses redefine certain steps of an algorithm without changing the algorithm's structure.(定义一个操作中的算法的框架,而将一些步骤延迟到子类中。使得子类可以不改变一个算法的结构即可重新定义该算法的某些特定步骤。)

模板方法模式的通用类图10-3:

10-3

模板方法确实非常简单,仅仅使用了java的继承机制,但它是一个应用非常广泛的模式。其中AbstractClass叫做抽象模板,他的方法分为两类:

  • 基本方法
    基本方法也叫作基本操作,是由子类实现的方法,并且在模板方法中被调用。也就是定义中说的将一些步骤延迟到子类,上面悍马例子中,几个抽象的方法,定义悍马的一些行为的方法。

  • 模板方法
    可以有一个或几个,一般是一个具体方法,实现对基本方法的调度,完成固定的逻辑。也就是例子中的run方法。

注意 为了防止恶意的操作,一般模板方法都加上final关键字,不允许被覆写。

在类图中还有 一个角色:具体模板,ConcreteClass1和ConcreteClass2,实现抽象模板中的基本方法。

注意 抽象模板中的基本方法尽量设计为protected类型,符合迪米特法则,不需要暴露的属性或方法尽量不要设置为public类型,实现类若非必要,尽量不要扩大父类中的访问权限

  1. 模板方法模式的应用

3.1模板方法模式的优点

  • 封装不变部分,扩展可变部分
    把认为不变的特定流程的算法封装到父类实现,而可变部分则可以通过继承来继续扩展。符合开闭原则,对修改关闭,对扩展开放。
  • 提取公共部分代码,便于维护
    我们例子中就是最好的证明,run方法中的流程固定不变,如果不提到父类中,那么美创建一种子类模板就得实现一次run,如果流程发生变动,那么每一个子类模板都需要维护。
  • 行为由父类控制,子类实现
    基本方法由子类实现,因此子类可以通过扩展的方式增加相应的功能,符合开闭原则。

3.2 模板方法模式的缺点
这个很明显,按照我们的设计习惯,抽象类负责声明最抽象、最一般的事物属性和方法,实现类完成具体的失误属性和方法。但这里确实是子类的实现方式对父类产生了影响,基本方法的实现是在子类中,但是流程执行方法是由父类的方法执行的。在复杂的项目中,会带来代码阅读的难度。

3.3模板方法模式的使用场景

  • 多个子类公有的方法,并且逻辑基本相同时。(这种思想应该一直落实在我们的开发中,重复的代码能提取为公共方法就尽量提取出来
  • 重复、复杂的算法,可以把核心算法设计为模板方法,周边相关的细节功能则由各个子类实现(这个做法在我们开发比较复杂的业务逻辑也非常有用,先将整个复杂逻辑细化为几个步骤,也就是几个具体实现方法,然后我们明确的确定流程,再去具体实现每个步骤的操作;这样做可以帮我们很好的理清思路
  • 重构时,模板方法模式是一个经常使用的模式,把相同的代码提取到父类中,然后通过钩子函数约束其行为(钩子函数是个啥具体往下看看是怎么解释的)
  1. 模板方法模式的扩展

到目前为止,例子中两个模型都稳定的运行,突然有一天,需求有了变化,H1的喇叭想让他响就响,H2型号的喇叭不要有声音,类图10-4:


10-4

类图的改动很小,就在HUmmerModel中增加了一个实现方法isAlarm,确定个型号的悍马是否需要声音,代码如下:

//抽象模板
public abstract class HummerModel {
    //能发动
    protected abstract void start();
    //停
    protected abstract void stop();
    //喇叭
    protected abstract void alarm();
    //引擎响
    protected abstract void engineBoom();
    //跑
    public final void run(){
        //先发动汽车
        this.start();
        //引擎开始轰鸣
        this.engineBoom();
        //按喇叭
        if(this.isAlarm()){
            this.alarm();
        }
        //停车
        this.stop();
    }
    protected boolean isAlarm(){
        return true;
    }
}
//具体模板1
public class HummerH1Model extends HummerModel {
    private boolean alarmFlag = true; //要响喇叭
    @Override
    protected void start() {
        System.out.println("悍马H1发动。。。");
    }

    @Override
    protected void stop() {
        System.out.println("悍马H1停车。。。");
    }

    @Override
    protected void alarm() {
        System.out.println("悍马H1鸣笛。。。");
    }

    @Override
    protected void engineBoom() {
        System.out.println("悍马H1引擎声音。。。");
    }

    @Override
    protected boolean isAlarm() {
        return this.alarmFlag;
    }
    public void setAlarmFlag(boolean alarmFlag){
        this.alarmFlag = alarmFlag;
    }
}
//具体模板2
public class HummerH2Model extends HummerModel {
    @Override
    protected void start() {
        System.out.println("悍马H2发动。。。");
    }

    @Override
    protected void stop() {
        System.out.println("悍马H2停车。。。");
    }

    @Override
    protected void alarm() {
        System.out.println("悍马H2鸣笛。。。");
    }

    @Override
    protected void engineBoom() {
        System.out.println("悍马H2引擎声音。。。");
    }

    @Override
    protected boolean isAlarm() {
        return false;
    }
}

H1型号的悍马是由客户控制是否要响喇叭,也就是说外界条件改变,影响到模板方法的执行。在抽象类中isAlarm的返回值就是影响了模板方法的执行结果,该方法就叫钩子方法(Hook Method)。(这个模板方法模式就是典型的子类方法侵入了父类,改变了父类的执行结果,其实这是有问题的,这也是模板方法的缺点吧。但是往后面看作者的解释,这个好像并不是多大的问题,反而认为模板方法模式的这种处理方式算是比较好的处理了父类依赖子类的场景。果然原则都不能学死,还是得看具体的情景

  1. 最佳实践

"父类怎么调用子类的方法"。这个问题很有普遍性,父类是否可以调用子类的方法呢?书中认为能,但是强烈、极度的不建议这么做,父类调用子类的方法如下:

  • 把子类传递到父类的有参构造中,然后调用。(这就是让父类直接依赖子类)
  • 使用反射的方式调用,你使用反射还有谁不能调用的?!
  • 父类调用子类的静态方法。(类方法都可以直接调用)
    但是这些都是不被允许的做法,父类建立框架,子类重写了父类的部分方法后,再调用从父类继承的方法,产生不同的结果,这是不是可以理解为父类调用了子类的方法,修改之类,影响父类的行为的结果。(这一段不是很理解,父类为什么要去调用子类的方法,哪怕是模板方法模式,看是声明的父类然后来执行,但是实际创建的实例肯定是子类吧
    这个模板方法模式应该是很好理解,并且很好用的模式,能很有效的帮助我们抽取重复的代码吧

内容来自《设计模式之禅》

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 204,590评论 6 478
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 86,808评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 151,151评论 0 337
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,779评论 1 277
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,773评论 5 367
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,656评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 38,022评论 3 398
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,678评论 0 258
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 41,038评论 1 299
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,659评论 2 321
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,756评论 1 330
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,411评论 4 321
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 39,005评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,973评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,203评论 1 260
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 45,053评论 2 350
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,495评论 2 343

推荐阅读更多精彩内容