17-1 模板方法模式讲解
17-2 模板方法模式coding
我们引入一个业务场景:制作视频课程,要制作ppt,还要去写上一个手记,还有提供一些前端的图片素材 ,像前端的课程就会提供图片素材,而后端的课程就不会提供图片素材,不同的课程的制作步骤,主线上是一致的,但是在细分上还是有一些不同,像手记并不是所有的课程都要写,有些课程需要写手记 ,而有些课程就不需要写手记,我们就可以把这个方法制作成一个勾子方法:
有一个抽象类:
public abstract class ACourse {
/** 声明成final就是不希望子类去覆盖这个方法 */
protected final void makeCourse() {
}
final void makePPT() {
System.out.println("制作PPT");
}
final void makeVideo() {
System.out.println("制作视频");
}
/** 这个编写手记是一个可选项 */
final void writeArticle() {
System.out.println("编写手记");
}
/** 我们就要给这个编写手记声明一个勾子方法 */
/** 这个并不是一个final方法,子类是可以进行覆盖的 */
protected boolean needWriteArticle() {
return false;
}
abstract void packageCourse();
}
Java设计模式的课程类:
public class DesignPatternCourse extends ACourse {
@Override
void packageCourse() {
System.out.println("提供课程Java源代码");
}
}
前端课程类:
public class FECourse extends ACourse {
@Override
void packageCourse() {
System.out.println("提供课程的前端代码");
System.out.println("提供课程的图片等多媒体素材");
}
}
现在的类图就是:
我们重新编写这个抽象类,编写制作课程的方法:
public abstract class ACourse {
/** 声明成final就是不希望子类去覆盖这个方法 */
protected final void makeCourse() {
this.makePPT();
this.makeVideo();
/** 这里是否需要写手记交由勾子方法来决定 */
if (needWriteArticle()) {
this.writeArticle();
}
this.packageCourse();
}
final void makePPT() {
System.out.println("制作PPT");
}
final void makeVideo() {
System.out.println("制作视频");
}
/** 这个编写手记是一个可选项 */
final void writeArticle() {
System.out.println("编写手记");
}
/** 我们就要给这个编写手记声明一个勾子方法 */
/** 这个并不是一个final方法,子类是可以进行覆盖的 */
protected boolean needWriteArticle() {
return false;
}
abstract void packageCourse();
}
我们来进行测试一下:
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--");
}
}
执行结果如下:
后端设计模式课程start–
制作PPT
制作视频
提供课程Java源代码
后端设计模式课程end–
前端式课程start–
制作PPT
制作视频
提供课程的前端代码
提供课程的图片等多媒体素材
前端课程end–
现在,我们的设计模式课程需要写手记,我们在这个类里面就是可以重写勾子方法:
public class DesignPatternCourse extends ACourse {
@Override
void packageCourse() {
System.out.println("提供课程Java源代码");
}
@Override
protected boolean needWriteArticle() {
return true;
}
}
执行结果:
后端设计模式课程start–
制作PPT
制作视频
编写手记
提供课程Java源代码
后端设计模式课程end–
前端式课程start–
制作PPT
制作视频
提供课程的前端代码
提供课程的图片等多媒体素材
前端课程end–
执行结果里面的设计模式的课程里面就有了编写手记的这个功能;
这个就是现在的类图:
现在,又一个需求:在前端课程里面,有angular课程,vue课程,有时候,我们的angular课程需要写手记,而vue课程不需要写手记,如果我们单单在FECourse里面重写勾子方法,那么所有的前端课程都要写手记 ,这就不符合需求了:
我们就是可以这样来做,定义一个是否要写手记的标识属性,通过构造器的方式来进行注入:
public class FECourse extends ACourse {
private boolean needWriteArticleFlag = false;
@Override
void packageCourse() {
System.out.println("提供课程的前端代码");
System.out.println("提供课程的图片等多媒体素材");
}
@Override
protected boolean needWriteArticle() {
return this.needWriteArticleFlag;
}
public FECourse(boolean needWriteArticleFlag) {
this.needWriteArticleFlag = needWriteArticleFlag;
}
}
如果想要前端课程写手记 ,我们就是可以这样来写:
public class Test {
public static void main(String[]args){
System.out.println("前端式课程start--");
ACourse feCourse = new FECourse(true);
feCourse.makeCourse();
System.out.println("前端课程end--");
}
}
执行结果如下:
前端式课程start–
制作PPT
制作视频
编写手记
提供课程的前端代码
提供课程的图片等多媒体素材
前端课程end–
如果不想写手记,我们就只需要把这个标识属性改成false就可以了:
public class Test {
public static void main(String[]args){
System.out.println("前端式课程start--");
ACourse feCourse = new FECourse(false);
feCourse.makeCourse();
System.out.println("前端课程end--");
}
}
执行结果如下:
前端式课程start–
制作PPT
制作视频
提供课程的前端代码
提供课程的图片等多媒体素材
前端课程end–
现在的类图为: