源码git地址 https://github.com/dlovetco/designMode
问题提出
老师在黑板上写了一套试卷,需要小明和小刚两个人做试卷。请用尽可能简洁的代码模拟出这个场景。
最简单无脑的做法
public class TemplateMode {
public static void main(String[] args) {
Ming ming = new Ming();
ming.answerQuestion1();
ming.answerQuestion2();
Gang gang = new Gang();
gang.answerQuestion1();
gang.answerQuestion2();
}
}
class Ming {
void answerQuestion1() {
System.out.println("问题一。。。。。。。。。。");
System.out.println("小明回答xxxx");
}
void answerQuestion2() {
System.out.println("问题二、、、、、、、、、、");
System.out.println("小明回答yyyy");
}
}
class Gang {
void answerQuestion1() {
System.out.println("问题一。。。。。。。。。。");
System.out.println("小刚回答aaaa");
}
void answerQuestion2() {
System.out.println("问题二、、、、、、、、、、");
System.out.println("小刚回答bbbb");
}
}
大家一看应该就能看出来,这代码丝毫没有结构,重复率高。那么我们改怎么修改这个代码呢?我们先分析一下这个问题就可以发现 问题的特点是问题都是一样的,不同的只是每个同学的回答。所以我们可以把题目抽象成父类,然后因为每个同学都要回答。把回答这个动作也抽象到父类中。
模板方法模式
package templateMode;
public class TemplateMode {
public static void main(String[] args) {
Ming ming = new Ming();
ming.question1();
ming.question2();
Gang gang = new Gang();
gang.question1();
gang.question2();
}
}
abstract class Exam{
void question1() {
System.out.println("问题一。。。。。。。。。。");
answerQuestion1();
}
abstract void answerQuestion1();
void question2() {
System.out.println("问题二。。。。。。。。。。");
answerQuestion2();
}
abstract void answerQuestion2();
}
class Ming extends Exam {
@Override
void answerQuestion1() {
System.out.println("小明回答xxxx");
}
@Override
void answerQuestion2() {
System.out.println("小明回答yyyy");
}
}
class Gang extends Exam {
@Override
void answerQuestion1() {
System.out.println("小刚回答aaaa");
}
@Override
void answerQuestion2() {
System.out.println("小刚回答bbbb");
}
}
这个模式比较特别一点的是抽象类和抽象方法的使用。因为对于每个学生来说都是看到题目,回答问题两个步骤。而且每个学生不一样的只有各自的回答,所以我们不光把试卷内容抽象出来,还把回答的流程也抽象出来。由于 回答 是交给学生做的,所以在父类中可以使用抽象方法来表明 回答 这一个动作。
模板方法模式所做的事情就是:把不变的事情移到父类,去除子类中的重复代码。
对了下面照例再贴一张plantuml
@startuml
abstract class Exam{
question1()
{abstract} answerQuestion1()
question2()
{abstract} answerQuestion2()
}
Exam <|--Ming
class Ming{
answerQuestion1()
}
Exam <|--Gang
class Gang{
answerQuestion2()
}
@enduml