设计模式之八:模板方法模式

泡咖啡和泡茶的共同点:

  1. 把水煮沸
  2. 沸水冲泡咖啡/茶叶
  3. 冲泡后的水倒入杯子
  4. 添加糖和牛奶/柠檬
cpp 复制代码
class CoffeineBeverage
{
public:
	void prepareRecipe()
	{
		boilWater();
		brew();
		pourInCup();
		addCondiments();
	}

private:
	void boilWater()
	{
		std::cout << "Boiling water" << std::endl;
	}

	virtual void brew() = 0;
	
	void pourInCup()
	{
		std::cout << "Pour in cup" << std::endl;
	}
	
	virtual void addCondiments() = 0;
};

在上面的代码种,prepareRecipe就是一个模板方法。

模板方法定义了一个算法的步骤,并允许子类为一个或多个步骤提供实现(依赖子类提供某些或所有步骤的实现,父类拥有并保护这个算法)。

模板方法模式在一个方法中定义了一个算法的骨架,而且将一些步骤延迟到子类中。模板方法使得子类可以在不改变算法结构的情况下,重新定义算法中的某些步骤。

cpp 复制代码
class CaffeineBeverage
{
public:
	void prepareRecipe()
	{
		boilWater();
		brew();
		pourInCup();

        // hook 用法1
        // 如果算法中的某个部分是可选的
        // 可以用hook来控制
        if(hook())
        {
		    addCondiments();
        }
        
        // hook 用法2
        // 让子类有机会对即将发生的,或
        // 刚刚发生的步骤做出反应。像是
        // 一种 callback
        hook();
	}

private:
	void boilWater()
	{
		std::cout << "Boiling water" << std::endl;
	}

	virtual void brew() = 0;
	
	void pourInCup()
	{
		std::cout << "Pour in cup" << std::endl;
	}
	
	virtual void addCondiments() = 0;

    void hook()
    {
        std::cout << "Hook" << std::endl;
    }
};

Hook(钩子)这个方法可以什么都不做,子类视情况决定是否覆盖它们,也可以做一些默认的事情。

钩子的存在可以让子类有能力对算法的不同点进行挂钩。

cpp 复制代码
if(customerWantsCondiments())
{
    addCondiments();
}

void addCondiments()
{
    // 子类可以自行决定什么时候需要添加调料
    return true; 
}

好莱坞原则:别调用我们,我们回调用你。此原则可以防止"依赖腐败",避免高层和低层组件之间有明显的环状依赖,...。(高层组件可以调用低层,如CaffeineBeverage调用Coffee和Tea的方法)(由超类控制一切,当它们需要的时候,自然会去调用子类)

模板方法模式对创建框架来说很常见,由框架控制如何做事情,而框架使用者来指定框架算法中每个步骤的细节。

  1. 用模板方法排序
java 复制代码
public static void sort(Object[] a)
{
	Object aux[] = (Object[])a.clone();
	// mergeSort方法包含排序算法,依赖于compareTo方法的实现
	mergeSort(aux, a, 0, a.length, 0);
}

private static void mergeSort(Object src[], Object desc[], int low, int high, int off)
{
	for (int i = low; i < high; i++)
	{
        // 需要实现compareTo方法
		for (int j = i; j > low; && ((Comparable)dest[j-1].compareTo((Comparable)dest[j]) >0); j++)
		{
            // swap是一个具体方法,已经在数组类中定义
			swap(dest, j, j - 1);
		}

	}
	return;
}

Java中的sort并不是真正定义在超类中,我们需要实现Comparable接口,实现compareTo方法。

java 复制代码
public class Duck implement Comparable
{
    String name;
    int weight;

    // ......
    // ......
    // ......

    public int compareTo(Object object)
    {
        Duck otherDucnk = (Duck)object;
        if(this.weight < otherDucnk.weight)
        {
            return -1;
        }
        else if ( ... ) { ... }
        else { ... }
    }
}
  • 模板方法的抽象类可以定义具体方法、抽象方法和钩子。
  • 策略模式和模板方法模式都封装算法,一个用组合,一个用继承。
  • 工厂方法是模板方法的一种特殊版本。
相关推荐
ThisIsMirror40 分钟前
设计模式简要介绍
设计模式
Lei活在当下6 小时前
【业务场景架构实战】7. 多代智能手表适配:Android APP 表盘编辑页的功能驱动设计
android·设计模式·架构
澄澈i8 小时前
设计模式学习[20]---桥接模式
c++·学习·设计模式·桥接模式
o0向阳而生0o10 小时前
106、23种设计模式之备忘录模式(15/23)
设计模式·备忘录模式
小猪佩奇TONY11 小时前
C++ 学习(3) ----设计模式
c++·学习·设计模式
zhulangfly1 天前
轻松理解智能体设计模式(1/6):提示链(Prompt Chaining)
设计模式·prompt chaining
da_vinci_x1 天前
2D角色动画进阶:Spine网格变形与序列帧特效的混合工作流
游戏·设计模式·设计师·photoshop·spine·游戏策划·游戏美术
代码萌新知2 天前
设计模式学习(五)装饰者模式、桥接模式、外观模式
java·学习·设计模式·桥接模式·装饰器模式·外观模式
charlie1145141912 天前
理解C++20的革命特性——协程支持2:编写简单的协程调度器
c++·学习·算法·设计模式·c++20·协程·调度器
笨手笨脚の2 天前
设计模式-适配器模式
设计模式·适配器模式·结构型设计模式