在讲模板模式原理前,我们按照管理,先来个生活中例子。茶馆需要开发一个自动的泡咖啡和泡茶的程序。

本文出自:《凯哥学设计模式》系列教程中的模板模式一

我们先来看看泡咖啡和泡茶的步骤:

document_image_rId9.png

我们根据上图写代码实现:

泡咖啡类,如下图:

document_image_rId10.png

泡茶类如下图:

document_image_rId11.png

测试类:

document_image_rId12.png

运行结果:

document_image_rId13.png

很简单。也很容易写出来。写出来很清楚。

从上图中,可以发现,两个流程几乎是一样的套路(步骤)。其中,不变的部分:水烧开、倒入杯子、送给客人。这三步是不变的。

变化的是:冲咖啡还是泡茶叶;加糖/牛奶还是加柠檬这两个步骤是变化的。

项目进化第一个版本:

我们将不变的抽取出来,放到一个公共的类中。HotDrink。然后让coffe和tea都继承公共的类。得到的类图如下:

document_image_rId14.png

hotdrink超类代码如下:

document_image_rId15.png

项目进化第二个版本:

经过分析,我们发现,两个流程的还有相同的地方:

1.两个流程的步骤都一样(都是五个步骤的);

2.无论泡茶还是泡咖啡都是brew操作;

3.无论加糖还是加柠檬都是添加调料的。

所以,我们对项目在进行一次提取:

我们将操作流程也提取到超类中,将2和3操作也放到超类中。让子类具体实现。所以得到类图如下:

document_image_rId16.png

我们来看看这次hotdrink类里面:

public abstract class HotDrink {

   public final void prepareRecipe(){

       boilWater();

       brew();

       pourInCup();

       addCondiments();

       send();


   }


   protected abstract void addCondiments();


   protected abstract void brew();


   private final void boilWater() {

       System.out.println("一.烧水");

   }

   private final void pourInCup() {

       System.out.println("三.倒入杯中");

   }

document_image_rId17.png

我们发现,在prepareRecipe方法和boilWter、pourInCup、send这四个操作都添加了final关键字来修改。这是为什么呢?

从上面分析,我们知道,都是五个步骤,而且五个步骤中的三个步骤(烧水、倒入杯中、送客人)也是固定不变的。那么,在Java中,固定不变的这个怎么表示呢?对了,就使用fianl这个关键字修饰就可以了。这样,就可以放置子类不能随便修改步骤(比如由五步变成三步),已经规定的不能在修改了。比如烧水这个不烧了,这样是不行的。

我们来看看,热饮coffee和tea的类:

document_image_rId18.png

hotDrinkTea:

document_image_rId19.png

测试方法:

document_image_rId20.png

运行结果:

document_image_rId21.png

我们对项目进化进行复盘总结,可以得到:

所谓的模板模式:封装了一个算法的步骤,并允许子类为一个或多个步骤方法提供实现。模板模式,可以使子类在不改变算法结构(如上面的五步)的情况下,重新定义算法中某些步骤(如上面的第二步和第四步)

模板模式类图如下:

document_image_rId22.png

类图说明:

1:是一个抽象类(如:hotDrink)

2:有个模板方法。这个模板方法是final的(如:prepareRecipe方法)

3:由三种方法:

AbsOperation:抽象的方法(泡咖啡、加牛奶)

concreteOp:具体的方法(如烧水。可以是final的也可以不是)

hook:钩子。可以选的子类可以覆盖父类的方法。

我们来演示下带有hook的。

比如,现在有了新需求,客户可以自己选择需不需要添加调料。这个怎么做呢?

本文来源:

凯哥Java(kaigejava)

凯哥个人博客:www.kaigejava.com


我们从新定义模板:

document_image_rId23.png

tea实现了该模板类,并且不加柠檬的:

document_image_rId24.png

测试运行:

document_image_rId25.png

结果:

document_image_rId26.png