一、定义
定义了一个算法的骨架,并允许子类为一个或多个步骤提供实现
模板方法使子类可以在不改变算法结构的情况下,重新定义算法的某些步骤
类型:行为型
二、适用场景
一次性实现一个算法的不变部分,并将可变的行为留给子类来实现
各子类中公共的行为被提取出来并集中到一个公共父类中,从而避免代码重复
三、优点
1、提高复用性
2、提高扩展性
3、符合开闭原则
四、缺点
1、类数目增加
2、增加了系统实现的复杂度
3、继承关系自身缺点,如果父类添加新的抽象方法,所有子类都要改一遍
五、模板方法-扩展
1、钩子方法
六、模板方法-相关设计模式
1、模板方法模式和工厂方法模式
工厂方法是模板方法的一种特殊实现。
2、模板方法模式和策略模式
策略模式:使不同的算法可以相互替换,并不影响客户处应用端的使用。可以改变算法的流程。如果代码中有大量的If-Else,可以考虑策略模式。
模板方法:针对定义一个算法的流程,将一些不太一样具体实现步骤交给子类实现。模板方法不改变算法的流程。
七、Coding
1、ACourse抽象类
public abstract class ACourse {
protected final void makeCourse(){
this.makePPT();
this.makeVideo();
if(needWrite()){
this.writeArticle();
}
this.packageCourse();
}
final void makePPT(){
System.out.println("制作PPT");
}
final void makeVideo(){
System.out.println("制作视频");
}
final void writeArticle(){
System.out.println("编写手记");
}
//钩子方法
protected boolean needWrite(){
return false;
}
abstract void packageCourse();
}
2、 DesignPatternCourse类
public class DesignPatternCourse extends ACourse {
@Override
void packageCourse() {
System.out.println("提供课程Java源代码");
}
@Override
protected boolean needWrite() {
return true;
}
}
3、FECourse类
/**
* 前端课程
*/
public class FECourse extends ACourse {
@Override
void packageCourse() {
System.out.println("提供课程前端源代码");
System.out.println("提供课程图片等多媒体素材");
}
}
4、测试
public class Test {
public static void main(String[] args) {
System.out.println("后端设计模式课程start---");
ACourse designPatternCourse = new DesignPatternCourse();
designPatternCourse.makeCourse();
System.out.println("后端设计模式课程end---");
System.out.println("前端设计模式课程start---");
ACourse feCourse = new FECourse();
feCourse.makeCourse();
System.out.println("前端设计模式课程end---");
}
}
5、输出结果
后端设计模式课程start--- 制作PPT 制作视频 编写手记 提供课程Java源代码 后端设计模式课程end--- 前端设计模式课程start--- 制作PPT 制作视频 提供课程前端源代码 提供课程图片等多媒体素材 前端设计模式课程end---
6、UML图

4、前端课程 控制needWrite
/**
* 前端课程
*/
public class FECourse extends ACourse {
private boolean needWriteArticleFlag = false;
@Override
void packageCourse() {
System.out.println("提供课程前端源代码");
System.out.println("提供课程图片等多媒体素材");
}
public void setNeedWriteArticleFlag(boolean needWriteArticleFlag) {
this.needWriteArticleFlag = needWriteArticleFlag;
}
@Override
protected boolean needWrite() {
return needWriteArticleFlag;
}
}