今天我们讲的模式,在平时运用过程中十分常见,那这个模式是什么呢,那就是模板方法模式。
让我们先来看一个简单的例子吧。
是不是感觉咖啡冲泡和茶冲泡的过程非常相似,如果设计这两个类,是不是也大致相同,那么我们可以一个超类,把相同的方法放在超类里面。
如果这样设计,子类中需要自己实现的两个方法虽然不一样,但是也挺相似的,都可以看成是冲泡和添加调料,那么我们把这两个方法抽象到超类中去试试看。
CaffeineBeverage类(超类):
public abstract class CaffeineBeverage { final void prepareRecipe() { //定义为final不希望子类覆盖这个方法 boilWate(); brew(); pourInCup(); addCondiments(); } abstract void brew(); //这两个方法定义为抽象,子类实现不同的做法 abstract void addCondiments(); void boilWate() { System.out.println("Boiling water"); } void pourInCup() { System.out.println("Pouring into cup"); } }Tea类: public class Tea extends CaffeineBeverage { public void brew() { System.out.println("Steeping the tea"); } public void addCondiments() { System.out.println("Adding Lemon"); } }Coffee类: public class Coffee extends CaffeineBeverage { public void brew() { System.out.println("Dripping coffee through filter"); } public void addCondiments() { System.out.println("Adding sugar and milk"); } }是不是很容易就明白了模板方法模式的使用,那么我们来看看他的定义吧。
模板方法模式:在一个方法中定义一个算法骨架,而将一些步骤延迟到子类中。模板方法使得子类可以在不改变算法结构的情况下,重新定义算法中的某些步骤。
模板方法类图:
在模板方法模式中还有一个知识点:钩子。
钩子?什么是钩子?
钩子是一种被声明在抽象类中的方法,可以让子类有能力对算法的不同点进行选择。
让我们来改变下之前我们写的超类吧:
public abstract class CaffeineBeverage { final void prepareRecipe() { //定义为final不希望子类覆盖这个方法 boilWate(); brew(); pourInCup(); if (customerWantsCondiments()) { addCondiments(); } } abstract void brew(); //这两个方法定义为抽象,子类实现不同的做法 abstract void addCondiments(); void boilWate() { System.out.println("Boiling water"); } void pourInCup() { System.out.println("Pouring into cup"); } boolean customerWantsCondiments() { return true; } }我们给之前的超类加了一个判断条件,这样就可以在子类中,通过实现判断条件中的方法来决定是否要使用某些方法。
模板方法模式我们就讲这些了,接下来又到了我们总结的时间了:
1:模板方法定义了算法的步骤,把这些步骤的实现延迟到子类。
2:模板方法模式为我们提供了一种代码复用的重要技巧。
3:模板方法的抽象类可以定义具体方法、抽象方法和钩子。
4:抽象方法由子类实现。
5:钩子是一种方法,它在抽象类中不做事,或者只做默认的事,子类可以选择要不要去覆盖它。
6:为了防止子类改变模板方法中的算法,可以将模板方法声明为final。
7:策略模式和模板方法模式都是封装算法,一个用组合,一个用继承。
8:工厂方法是模板方法的一种特殊版本。
