7.设计模式-模板方法模式

定义:定义一个操作中的算法的骨架,而将一些步骤延迟到子类中。模板方法使得子类可以不改变一个算法的结构即可重定义该算法的某些特定步骤

需求

实现一个抄写试卷题目的程序

需求分析

  • 识别出变化的内容,只将变化定义在子类,其他定义在父类

代码实现

模板类

c 复制代码
#include <stdio.h>
#include <stdlib.h>
#include <string.h>

#define PAPERMAXQUESTION 10
typedef struct TestPaper {
    char *testQuestion[PAPERMAXQUESTION];
    char *answers[PAPERMAXQUESTION];
    int questionNum;
    int answerNum;
    void (*SetQuestion)(struct TestPaper *);
    void (*PrintQuestion)(struct TestPaper *);
    void (*AddQuestion)(struct TestPaper *, char *);
    // 在模板类不定义
    void (*SetAnswer)(struct TestPaper *);
}TestPaper;

void AddQuestion(struct TestPaper *obj, char *question) {
    obj->testQuestion[(obj->questionNum)++] = question;
    return;
}
void SetQuestion(TestPaper *obj) {
    obj->AddQuestion(obj, "杨过得到,后来给了郭靖,炼成倚天剑、屠龙刀的玄铁可能是[ ] \na.球磨铸铁 b.马口铁 c.高速合金钢 d.碳素纤维");
    obj->AddQuestion(obj, "杨过、程英、陆无双铲除了情花,造成[ ] \na.使这种植物不再害人 b.使一种珍稀物种灭绝 c.破坏了那个生物圈的生态平衡 d.造成该地区沙漠化");
    obj->AddQuestion(obj, "蓝凤凰致使华山师徒、桃谷六仙呕吐不止,如果你是大夫,会给他们开什么药[ ]\na.阿司匹林 b.牛黄解毒片 c.氟哌酸 d.让他们喝大量的生牛奶 e.以上全不对");
    return;
}
void PrintQuestion(TestPaper *obj) {
    printf("===============================================\n");
    printf("试验:%d道试题\n", obj->questionNum);
    for(int i=0; i<obj->questionNum; i++) {
        printf("题目%d:%s\n", i+1, obj->testQuestion[i]);
        printf("答案:%s\n", obj->answers[i]);
    }
}

TestPaper *InitTestPaper() {
    TestPaper *obj = (TestPaper *)malloc(sizeof(TestPaper));
    obj->PrintQuestion = PrintQuestion;
    obj->AddQuestion = AddQuestion;
    obj->SetQuestion = SetQuestion;
    obj->questionNum = 0;
    obj->questionNum = 0;
    memset(obj->testQuestion, 0, sizeof(char *)*PAPERMAXQUESTION);
    memset(obj->answers, 0, sizeof(char *)*PAPERMAXQUESTION);
    obj->SetQuestion(obj);
    return obj;
}

具体类

c 复制代码
typedef struct TestPaperA {
    TestPaper base;
}TestPaperA;
void PaperASetAnswer(TestPaper *obj) {
    obj->answers[obj->answerNum++] = "b";
    obj->answers[obj->answerNum++] = "b";
    obj->answers[obj->answerNum++] = "b";
    return;
}
TestPaperA *InitTestPaperA() {
    TestPaperA *obj = (TestPaperA *)malloc(sizeof(TestPaperA));
    TestPaper *tmp = InitTestPaper();
    obj->base = *tmp;
    free(tmp);
    // 子类只定义一个设置答案接口
    obj->base.SetAnswer = PaperASetAnswer;
    obj->base.SetAnswer((TestPaper *)obj);
    return obj;
}
typedef struct TestPaperB {
    TestPaper base;
}TestPaperB;
void PaperBSetAnswer(TestPaper *obj) {
    obj->answers[obj->answerNum++] = "c";
    obj->answers[obj->answerNum++] = "c";
    obj->answers[obj->answerNum++] = "c";
    return;
}
TestPaperB *InitTestPaperB() {
    TestPaperB *obj = (TestPaperB *)malloc(sizeof(TestPaperB));
    TestPaper *tmp = InitTestPaper();
    obj->base = *tmp;
    free(tmp);
    obj->base.SetAnswer = PaperBSetAnswer;
    obj->base.SetAnswer((TestPaper *)obj);
    return obj;
}

客户端

c 复制代码
int main() {
    TestPaperA *paperA = InitTestPaperA();
    ((TestPaper *)paperA)->PrintQuestion((TestPaper *)paperA);
    TestPaperB *paperB = InitTestPaperB();
    ((TestPaper *)paperB)->PrintQuestion((TestPaper *)paperB);
    return 0;
}

UML图

总结

  • 模板方法模式应用场景?当不变的和可变的行为在方法的子类实现中混合在一起的时候,不变的行为就会在子类中重复出现。我们通过模板方法模式把这些行为搬移到单一的地方,这样就帮助子类摆脱重复的不变行为的纠缠。
相关推荐
数智工坊18 小时前
【UniT论文阅读】:用统一物理语言打通人类与人形机器人的知识壁垒
论文阅读·人工智能·深度学习·算法·机器人
梓䈑18 小时前
【算法题攻略】模拟
c++·算法
Evand J18 小时前
【课题推荐与代码介绍】卡尔曼滤波器正反向估计算法原理与MATLAB实现
开发语言·算法·matlab
DFT计算杂谈19 小时前
VASP新手入门: IVDW 色散修正参数
linux·运维·服务器·python·算法
吃着火锅x唱着歌19 小时前
LeetCode 962.最大宽度坡
算法·leetcode·职场和发展
无限进步_19 小时前
【C++】C++11的类功能增强与STL变化
java·前端·数据结构·c++·后端·算法
WL_Aurora19 小时前
Python 算法基础篇之排序算法(一):冒泡、选择、插入
python·算法·排序算法
凌波粒19 小时前
LeetCode--257. 二叉树的所有路径(二叉树)
算法·leetcode·职场和发展
AI算法沐枫19 小时前
大一学生如何入门机器学习,深度学习,学习顺序如何?
人工智能·python·深度学习·学习·线性代数·算法·机器学习
codealy19 小时前
Rust 核心理论: 高并发与异步(三)
算法·rust