模板方法详解

1.基本介绍

  1. 模板方法模式(Template Method Pattern),又叫模板模式(Template Pattern), 在一个抽象类公开定义了执行 它的方法的模板。它的子类可以按需要重写方法实现,但调用将以抽象类中定义的方式进行。
  2. 简单说,模板方法模式 定义一个操作中的算法的骨架,而将一些步骤延迟到子类中,使得子类可以不改变一 个算法的结构,就可以重定义该算法的某些特定步骤
  3. 这种类型的设计模式属于行为型模式

2.原理类图

3.具体需求

1.豆浆制作

编写制作豆浆的程序,说明如下:

  • 制作豆浆的流程:选材--->添加配料--->浸泡--->放到豆浆机打碎
  • 通过添加不同的配料,可以制作出不同口味的豆浆
  • 选材、浸泡和放到豆浆机打碎这几个步骤对于制作每种口味的豆浆都是一样的

2. 模板方法模式方案

3.具体实现

java 复制代码
// 抽象类,表示豆浆	SoyaMilk.java
public abstract class SoyaMilk {
	// 模板方法:可以做成final,不让子类去覆盖
	final void make() {
		select();
		addCondiment();
		soak();
		beat();
	}
	
	//选材料
	void select() { System.out.println("第一步:选择新鲜的豆子"); }
	//添加不同的配料:抽象方法,由子类具体实现
	abstract void addCondiment();
	//浸泡
	void soak() { System.out.println("第三步:豆子和配料开始浸泡3H"); }
	//榨汁
	void beat() { System.out.println("第四步:豆子和配料放入豆浆机榨汁"); }
}

// RedBeanSoyaMilk.java
public class ReadBeanSoyaMilk extends SoyaMilk {
	@Override
	void addCondiment() {
		System.out.println("第二步:加入上好的红豆");
	}
}

// PeanutSoyMilk.java
public class PeanutSoyaMilk extends SoyaMilk {
	@Override
	void addCondiment() {
		System.out.println("第二步:加入上好的花生");
	}
}

// Client.java
public class Client {
	public static void main(String[] args) {
		System.out.println("=======制作红豆豆浆=======");
		SoyaMilk redBeanSoyaMilk = new ReadBeanSoyaMilk();
		redBeanSoyaMilk.make();
		
		System.out.println("=======制作花生豆浆=======");
		SoyaMilk peanutSoyaMilk = new PeanutSoyaMilk();
		peanutSoyaMilk.make();
	}
}

4.模板方法模式的钩子方法

在模板方法模式的父类中,可以定义一个方法,它默认不做任何事,子类可以视情况要不要覆盖它,该方法称为"钩子"。还是用上面做豆浆的例子来讲解,比如,我们还希望制作纯豆浆,不添加任何的配料,请使用钩子方法对前面的模板方法进行改造。代码实现如下:

java 复制代码
// RedBeanSoyaMilk.java/PeanutSoyaMilk.java同上,略

//抽象类,表示豆浆,SoyaMilk
public abstract class SoyaMilk {

	//模板方法:可以做成final,不让子类去覆盖
	final void make() {
		select();
		if(customerWantCondiment()) {
			addCondiment();
		}	
		soak();
		beat();
	}

	//1.选材料
	void select() { System.out.println("第一步:选择新鲜的豆子"); }
	//2.添加不同的配料:抽象方法,由子类具体实现
	abstract void addCondiment();
	//3.浸泡
	void soak() { System.out.println("第三步:豆子和配料开始浸泡3H"); }
	//4.榨汁
	void beat() { System.out.println("第四步:豆子和配料放入豆浆机榨汁"); }

	//钩子方法:决定是否需要添加配料
	boolean customerWantCondiment() {
		return true;//默认情况下是要加配料的
	}
}

// PureSoyaMilk.java
public class PureSoyaMilk extends SoyaMilk {
	@Override
	void addCondiment() {
		// 添加配料的方法 空实现 即可
	}
	@Override
	boolean customerWantCondiment() {
		return false;
	}
}

// Client.java
public class Client {
	public static void main(String[] args) {
		System.out.println("=制作纯豆浆=");
		SoyaMilk pureSoyMilk = new PureSoyaMilk();
		pureSoyMilk.make();
	}
}

4.在spring中的应用

我们先看一串代码:

可以看到ConfigurableApplicationContext对象的运行类型为AnnotationConfigApplicationContext

观察类图:

经过源码分析可以得知:

5.注意事项和细节

  1. 基本思想是:算法只存在于一个地方,也就是在父类中,容易修改。需要修改算法时,只要修改父类的模板方 法或者已经实现的某些步骤,子类就会继承这些修改;
  2. 实现了最大化代码复用。父类的模板方法和已实现的某些步骤会被子类继承而直接使用;
  3. 既统一了算法,也提供了很大的灵活性。父类的模板方法确保了算法的结构保持不变,同时由子类提供部分步骤的实现;
  4. 该模式的不足之处:每一个不同的实现都需要一个子类实现,导致类的个数增加,使得系统更加庞大;
  5. 一般模板方法都加上 final 关键字, 防止子类重写模板方法;
    会被子类继承而直接使用;
  6. 既统一了算法,也提供了很大的灵活性。父类的模板方法确保了算法的结构保持不变,同时由子类提供部分步骤的实现;
  7. 该模式的不足之处:每一个不同的实现都需要一个子类实现,导致类的个数增加,使得系统更加庞大;
  8. 一般模板方法都加上 final 关键字, 防止子类重写模板方法;
  9. 模板方法模式使用场景:当要完成在某个过程,该过程要执行一系列步骤 ,这一系列的步骤基本相同,但其个别步骤在实现时可能不同,通常考虑用模板方法模式来处理.
相关推荐
成钰9 小时前
设计模式之单例模式:一个类就只有一个实例
单例模式·设计模式
o0向阳而生0o10 小时前
110、23种设计模式之状态模式(19/23)
设计模式·状态模式
_院长大人_10 小时前
设计模式-单例模式
单例模式·设计模式
崎岖Qiu19 小时前
【设计模式笔记17】:单例模式1-模式分析
java·笔记·单例模式·设计模式
安冬的码畜日常1 天前
【JUnit实战3_27】第十六章:用 JUnit 测试 Spring 应用:通过实战案例深入理解 IoC 原理
spring·观察者模式·设计模式·单元测试·ioc·依赖注入·junit5
她说彩礼65万1 天前
C#设计模式 单例模式实现方式
单例模式·设计模式·c#
安冬的码畜日常1 天前
【JUnit实战3_28】第十七章:用 JUnit 5 实测 SpringBoot 项目
spring boot·功能测试·测试工具·设计模式·单元测试·junit5
围巾哥萧尘1 天前
TRAE Agent 歌曲创作助手构建与使用教程🧣
设计模式
superman超哥1 天前
仓颉语言中流式I/O的设计模式深度剖析
开发语言·后端·设计模式·仓颉
m0_748248021 天前
Spring设计模式刨根问底
java·spring·设计模式