模板模式,定义一个操作中算法的骨架,而将一些步骤延迟到子类中,模板方法使得子类可以不改变算法的结构即可重定义该算法的某些特定步骤。
简单的讲,完成一件事情,有固定的数个步骤,但是每个步骤根据对象的不同,而实现细节不同;就可以在父类中定义一个完成该事情的总方法,按照完成事件需要的步骤去调用其每个步骤的实现方法。每个步骤的具体实现,由子类完成。
抽象父类(AbstractClass):实现了模板方法,定义了算法的骨架。
具体类(ConcreteClass):实现抽象类中的抽象方法,即不同的对象的具体实现细节。
举个例子,炒菜可以分为三个步骤 :(1)备料 (2)炒菜 (3)盛菜端给客人享用。这三部就是算法的骨架 ;然而做不同菜需要的料,做的方法,以及如何盛装给客人享用都是不同的这个就是不同的实现细节。UML设计图如下:
1)抽象模板 Dish 类:
package cn.jaa.template_pattern;
/**
* @Author: Jaa
* @Description:
* @Date 2023/12/3 17:29
*/
public abstract class Dish {
public void doDish() {
this.preparation();
this.doing();
this.carriedDishes();
}
// 备菜
public abstract void preparation();
// 炒菜
public abstract void doing();
// 上菜
public abstract void carriedDishes();
}
2)EggsWithTomato 和 PepperWithPork 实现父类中的抽象方法:
package cn.jaa.template_pattern;
import lombok.extern.slf4j.Slf4j;
/**
* @Author: Jaa
* @Description: 番茄炒鸡蛋
* @Date 2023/12/3 17:35
*/
@Slf4j
public class EggsWithTomato extends Dish {
@Override
public void preparation() {
log.info("洗切番茄,打鸡蛋,开火倒油 ...");
}
@Override
public void doing() {
log.info("倒入鸡蛋炒黄,加番茄加盐一起炒匀 ...");
}
@Override
public void carriedDishes() {
log.info("将炒好的 EggsWithTomato 放入盘,端给客人 ...");
}
}
package cn.jaa.template_pattern;
import lombok.extern.slf4j.Slf4j;
/**
* @Author: Jaa
* @Description: 辣椒炒肉
* @Date 2023/12/3 17:41
*/
@Slf4j
public class PepperWithPork extends Dish {
@Override
public void preparation() {
log.info("切猪肉,辣椒,准备配料 ...");
}
@Override
public void doing() {
log.info("倒油红锅,加小料炒香放入pork炒熟加配料,炒匀 ...");
}
@Override
public void carriedDishes() {
log.info("将炒好的 PepperWithPork 出锅放入盘,端给客人 ...");
}
}
3)测试模板模式:
package cn.jaa.template_pattern;
import lombok.extern.slf4j.Slf4j;
/**
* @Author: Jaa
* @Description:
* @Date 2023/12/3 17:39
*/
@Slf4j
public class TemplateDemoTest {
public static void main(String[] args) {
Dish eggsWithTomato = new EggsWithTomato();
eggsWithTomato.doDish();
log.info("-------------------------");
Dish pepperWithPork = new PepperWithPork();
pepperWithPork.doDish();
}
}
打印结果: