模板方法模式(Template Method Pattern)——复杂流程步骤的设计

模式概述

在现实生活中,很多事情都包含几个实现步骤,例如请客吃饭,无论吃什么,一般都包含点单、吃东西、买单等几个步骤,通常情况下这几个步骤的次序是:点单 –> 吃东西 –> 买单。

在这三个步骤中,点单和买单大同小异,最大的区别在于第二步——吃什么?不同的人、不同的地方有不同的饮食习惯。如果站在软件工程的角度来看,这就是变化的地方,也可以说是可扩展的点。

软件开发中,经常会遇到类似的情况,某个方法的实现需要多个步骤(类似“请客”),其中有些步骤是固定的(类似“点单”和“买单”),而有些步骤并不固定,存在可变性(类似“吃东西”)。

为了提高代码的复用性和系统的灵活性,可以使用一种称之为模板方法模式的设计模式来对这类情况进行设计。

在模板方法模式中,将实现功能的每一个步骤所对应的方法称为基本方法(例如“点单”、“吃东西”和“买单”),而调用这些基本方法同时定义基本方法的执行次序的方法称为模板方法(例如“请客”)。

在模板方法模式中,可以将相同的代码放在父类中,例如将模板方法“请客”以及基本方法“点单”和“买单”的实现放在父类中,而对于基本方法“吃东西”,在父类中只做一个声明,将其具体实现放在不同的子类中,在一个子类中提供“吃面条”的实现,而另一个子类提供“吃米饭”的实现。

通过使用模板方法模式,一方面提高了代码的复用性,另一方面还可以利用面向对象的多态性,在运行时选择一种具体子类,实现完整的“请客”方法,提高系统的灵活性和可扩展性。

模式定义

模板方法模式定义如下:

模板方法模式(Template Method Pattern):定义一个操作中算法的框架,而将一些步骤延迟到子类中。模板方法模式使得子类可以不改变一个算法的结构即可重定义该算法的某些特定步骤。

模板方法模式是一种基于继承的代码复用技术,它是一种类行为型模式。

模板方法模式是结构最简单的行为型设计模式,在其结构中只存在父类与子类之间的继承关系。通过使用模板方法模式,可以将一些复杂流程的实现步骤封装在一系列基本方法中,在抽象父类中提供一个称之为模板方法的方法来定义这些基本方法的执行次序,而通过其子类来覆盖某些步骤,从而使得相同的算法框架可以有不同的执行结果。

模板方法模式提供了一个模板方法来定义算法框架,而某些具体步骤的实现可以在其子类中完成。

模式结构图

模板方法模式结构比较简单,其核心是抽象类和其中的模板方法的设计,其结构如下图所示:

模板方法模式包含如下两个角色:

  • AbstractClass(抽象类):在抽象类中定义了一系列基本操作(Primitive Operations),这些基本操作可以是具体的,也可以是抽象的,每一个基本操作对应算法的一个步骤,在其子类中可以重定义或实现这些步骤。同时,在抽象类中实现了一个模板方法(Template Method),用于定义一个算法的框架,模板方法不仅可以调用在抽象类中实现的基本方法,也可以调用在抽象类的子类中实现的基本方法,还可以调用其他对象中的方法。
  • ConcreteClass(具体子类):它是抽象类的子类,用于实现在父类中声明的抽象基本操作以完成子类特定算法的步骤,也可以覆盖在父类中已经实现的具体基本操作。

模式伪代码

抽象父类AbstractClass负责给出一个算法的轮廓和框架,如下:

/**
 * 抽象模板
 */
public abstract class AbstractClass {

    /**
     * 具体方法1
     */
    public void method1() {

    }

    /**
     * 抽象方法2
     */
    public abstract void method2();

    /**
     * 具体方法3
     */
    public void method3() {
        
    }

    /**
     * 模板方法
     */
    public void templateMethod() {
        method1();
        method2();
        method3();
    }
}

其中实现这些具体逻辑步骤的方法即为基本方法,而将这些基本方法汇总起来的方法即为模板方法

子类ConcreteClass负责给出这个算法的各个逻辑实现,如下:

public class ConcreteClass extends AbstractClass {

    @Override
    public void method2() {
        // 抽象步骤的实现
    }

    @Override
    public void method3() {
        // 也可覆盖父类中已经实现的具体方法
        super.method3();
    }
}

这样,就可以通过扩展不同的子类来覆盖某些步骤,从而使得相同的算法框架可以有不同的执行结果。

模式总结

写代码的一个很重要的思考点就是变与不变,程序中哪些功能是可变的,哪些功能是不变的,我们可以把不变的部分提取出来,进行公共的实现,把变化的部分分离出来,用接口来封装隔离变化,或用抽象类约束子类行为。模板方法模式就很好的体现了这一点。

模板方法模式是基于继承的代码复用技术,它体现了面向对象的诸多重要思想,是一种使用较为频繁的模式。模板方法模式广泛应用于框架设计中,以确保通过父类来控制处理流程的逻辑顺序(如框架的初始化,测试流程的设置等)。

主要优点

  • 在父类中形式化地定义一个算法,而由它的子类来实现细节的处理,在子类实现详细的处理算法时并不会改变算法中步骤的执行次序
  • 模板方法模式是一种代码复用技术,它提取了公共行为,将公共行为放在父类中,而通过其子类来实现不同的行为,它鼓励我们恰当使用继承来实现代码复用
  • 可实现一种反向控制结构,通过子类覆盖父类的方法来决定某一特定步骤是否需要执行
  • 通过子类来覆盖父类的基本方法,不同的子类可以提供基本方法的不同实现,更换和增加新的子类很方便,符合单一职责原则和开闭原则

主要缺点

需要为每一个基本方法的不同实现提供一个子类,如果父类中可变的基本方法太多,将会导致类的个数增加,系统更加庞大,设计也更加抽象,此时,可结合 桥接模式 来进行设计。

适用场景

  • 对一些复杂的算法进行分割,将其算法中固定不变的部分设计为模板方法和父类具体方法,而一些可以改变的细节由其子类来实现
  • 各子类中公共的行为应被提取出来并集中到一个公共父类中
  • 需要通过子类来决定父类算法中某个步骤是否执行,实现子类对父类的反向控制

参考书籍:

《设计模式的艺术之道(软件开发人员内功修炼之道)》—— 刘伟