C++设计模式之 模板方法模式

news2025/1/16 16:59:10

【声明】本题目来源于卡码网(题目页面 (kamacoder.com))

【提示:如果不想看文字介绍,可以直接跳转到C++编码部分】


【设计模式大纲】


【简介】

        --什么是模板方法模式(第18种设计模式)

        模板方法模式(Template Method Pattern)是⼀种行为型设计模式, 它定义了⼀个算法的骨架,将⼀些步骤的实现延迟到⼦类。模板方法模式使得⼦类可以在不改变算法结构的情况下,重新定义算法中的某些步骤。

        举个简单的例⼦,做⼀道菜通常都需要包含⾄少三步:

  • 准备⻝材
  • 亨饪过程
  • 上菜

        不同菜品的亨饪过程是不⼀样的,但是我们可以先定义⼀个”⻣架”,包含这三个步骤,亨饪过程的过程放到具体的炒菜类中去实现,这样,⽆论炒什么菜,都可以沿⽤相同的炒菜算法,只需在⼦类中实现具体的炒菜步骤,从而提⾼了代码的复⽤性。


【基本结构】

        模板⽅法模式的基本结构包含以下两个角色:

  • 模板类AbstractClass :由⼀个模板⽅法和若⼲个基本⽅法构成,模板⽅法定义了逻辑的⻣架,按照顺序调⽤包含的基本⽅法,基本⽅法通常是⼀些抽象⽅法,这些⽅法由⼦类去实现。基本⽅法还包含⼀些具体⽅法,它们是算法的⼀部分但已经有默认实现,在具体⼦类中可以继承或者重写。
  • 具体类ConcreteClass :继承⾃模板类,实现了在模板类中定义的抽象⽅法,以完成算法中特定步骤的具体实现。

【简易实现】

        模板⽅法模式的简单示例如下:先以Java代码作以说明

1. 定义模板类,包含模板⽅法,定义了算法的⻣架, ⼀般都加上final 关键字C++为 const关键字),避免子类重写。

// 模板类
abstract class AbstractClass {
    // 模板⽅法,定义了算法的⻣架
    public final void templateMethod() {
        step1();
        step2();
        step3();
    }
    // 抽象⽅法,由⼦类实现
    protected abstract void step1();
    protected abstract void step2();
    protected abstract void step3();
}

2. 定义具体类, 实现模板类中的抽象⽅法

// 具体类
class ConcreteClass extends AbstractClass {
    @Override
    protected void step1() {
        System.out.println("Step 1 ");
    }
    @Override
    protected void step2() {
        System.out.println("Step 2 ");
    }
    @Override
    protected void step3() {
        System.out.println("Step 3");
    }
}

3. 客户端实现

/**
* @version Copyright (c) 2024 NCDC, Servo。 Unpublished - All rights reserved
* @file TemplateMethodMode.hpp
* @brief 模板方法模式
* @autor 写代码的小恐龙er
* @date 2024/01/20
*/

public class Main {
    public static void main(String[] args) {
        AbstractClass concreteTemplate = new ConcreteClass();
        // 触发整个算法的执⾏
        concreteTemplate.templateMethod();
    }
}

【应用场景】

        模板⽅法模式将算法的不变部分被封装在模板⽅法中,而可变部分算法由⼦类继承实现,这样做可以很好的提⾼代码的复⽤性,但是当算法的框架发⽣变化时,可能需要修改模板类,这也会影响到所有的⼦类。
        总体来说,当算法的整体步骤很固定,但是个别步骤在更详细的层次上的实现可能不同时,通常考虑模板⽅法模式来处理。在已有的⼯具和库中, Spring框架中的JdbcTemplate 类使⽤了模板⽅法模式,其中定义了⼀些执⾏数据库操作的模板⽅法,具体的数据库操作由回调函数提供。⽽在Java的JDK源码中, AbstractList 类也使⽤了模板⽅法模式,它提供了⼀些通用的⽅法,其中包括⼀些模板⽅法。具体的列表操作由⼦类实现。


【C++编码部分】

1. 题目描述

        小明喜欢品尝不同类型的咖啡,她发现每种咖啡的制作过程有一些相同的步骤,他决定设计一个简单的咖啡制作系统,使用模板方法模式定义咖啡的制作过程。系统支持两种咖啡类型:美式咖啡(American Coffee)和拿铁(Latte)。

        咖啡制作过程包括以下步骤:

        1. 研磨咖啡豆 Grinding coffee beans

        2. 冲泡咖啡 Brewing coffee

        3. 添加调料 Adding condiments

        其中,美式咖啡和拿铁的调料添加方式略有不同, 拿铁在添加调料时需要添加牛奶Adding milk。

2. 输入描述

        多行输入,每行包含一个数字,表示咖啡的选择(1 表示美式咖啡,2 表示拿铁)。

3. 输出描述

        根据每行输入,输出制作咖啡的过程,包括咖啡类型和各个制作步骤,末尾有一个空行。

4. C++编码实例

/**
* @version Copyright (c) 2024 NCDC, Servo。 Unpublished - All rights reserved
* @file TemplateMethodMode.hpp
* @brief 模板方法模式
* @autor 写代码的小恐龙er
* @date 2024/01/20
*/

#include <iostream>
#include <string>
#include <vector>

using namespace std;

// 前置声明

// 抽象模板类 -- 包含模板⽅法,定义了算法的⻣架, ⼀般都加上final 关键字,避免⼦类重写
class CoffeeMaker;
// 实现具体类 -- 美式
class AmericanCoffeeMaker;
// 实现具体类 -- 拿铁
class LatteCoffeeMaker;

// 类的定义

// 抽象模板类
class CoffeeMaker
{
// 成员接口函数
public:
    // 框架接口函数
    const void TemplateCoffeeMethod(){
        // 步骤一
        GrindingCoffeeBeans();
        // 步骤二
        BrewingCoffee();
        // 步骤三
        AddingCondiments();
        // 空行
        std::cout << endl;
    }
    // 研磨咖啡豆工序
    virtual void GrindingCoffeeBeans() = 0;
    // 冲泡咖啡工序
    virtual void BrewingCoffee() = 0;
    // 添加调料工序
    virtual void AddingCondiments(){
        std::cout << "Adding condiments" << endl;
    }
};

// 实现具体类 -- 美式
class AmericanCoffeeMaker : public CoffeeMaker
{
// 成员数据
private:
    string _coffeeName;

// 成员接口函数重载
public:
    // 构造函数
    AmericanCoffeeMaker(string name){
        this->_coffeeName = name;
        std::cout<< "Making " << _coffeeName << ":"<<endl;
    }
    // 研磨咖啡豆工序 重载
    void GrindingCoffeeBeans() override {
        std::cout << "Grinding coffee beans" << endl;
    }
    // 冲泡咖啡工序 重载
    void BrewingCoffee() override {
        std::cout << "Brewing coffee" << endl;
    }
};

// 实现具体类 -- 拿铁
class LatteCoffeeMaker : public CoffeeMaker
{
// 成员数据
private:
    string _coffeeName;

// 成员接口函数重载
public:
    // 构造函数
    LatteCoffeeMaker(string name){
        this->_coffeeName = name;
        std::cout<< "Making " << _coffeeName << ":"<<endl;
    }
    // 研磨咖啡豆工序 重载
    void GrindingCoffeeBeans() override {
        std::cout << "Grinding coffee beans" << endl;
    }
    // 冲泡咖啡工序 重载
    void BrewingCoffee() override {
        std::cout << "Brewing coffee" << endl;
    }
    // 添加调料工序 重载
    void AddingCondiments() override {
        std::cout << "Adding milk" << endl;
        std::cout << "Adding condiments" << endl;
    }
};


// 客户端代码
int main()
{
    // 输入的咖啡制作类型
    int type = 0;
    // 构造抽象模板类
    CoffeeMaker *coffee = nullptr;
    while(std::cin >> type){
        if(type == 1){
            coffee = new AmericanCoffeeMaker("American Coffee");
        }
        else if(type == 2){
            coffee = new LatteCoffeeMaker("Latte");
        }
        else break;
        
        // 开始统一制作咖啡
        coffee->TemplateCoffeeMethod();
    }
    
    // 析构
    if(coffee != nullptr){
        delete coffee;
        coffee = nullptr;
    }
    
    return 0;
}



......

To be continued.

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/1400864.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

大数据安全 | 期末复习(上)| 补档

文章目录 &#x1f4da;概述⭐️&#x1f407;大数据的定义、来源、特点&#x1f407;大数据安全的含义&#x1f407;大数据安全威胁&#x1f407;保障大数据安全&#x1f407;采集、存储、挖掘环节的安全技术&#x1f407;大数据用于安全&#x1f407;隐私的定义、属性、分类、…

Transformer 可解释性论文整理(超级详细)

Transformer 可解释性论文整理 前段时间想进一步的了解transformer的工作原理&#xff0c;于是找到了几篇可解释性的文章进行阅读&#xff0c;发现了许多比较有趣的现象和结论&#xff0c;对每篇文章都有自己的深度思考和理解&#xff0c;在此记录&#xff0c;欢迎交流。 1. …

「优选算法刷题」:二分查找

一、题目 给定一个 n 个元素有序的&#xff08;升序&#xff09;整型数组 nums 和一个目标值 target &#xff0c;写一个函数搜索 nums 中的 target&#xff0c;如果目标值存在返回下标&#xff0c;否则返回 -1。 示例 1: 输入: nums [-1,0,3,5,9,12], target 9 输出: 4 …

【JavaEE进阶】实现验证码

文章目录 &#x1f332;实现说明&#x1f343;Kaptcha插件介绍&#x1f6a9;插件原理&#x1f6a9;引入依赖&#x1f6a9;⽣成验证码&#x1f6a9;Kaptcha详细配置 &#x1f340;准备⼯作&#x1f334;约定前后端交互接⼝&#x1f6a9;需求分析&#x1f6a9;接⼝定义&#x1f6…

力扣 第 122 场双周赛 解题报告 | 珂学家 | 脑筋急转弯 + 滑窗反悔堆

前言 整体评价 倒开差点崩盘&#xff0c;T4这个反悔堆写吐了&#xff0c;T3往众数上去猜了&#xff0c;幸好case良心。 T1. 将数组分成最小总代价的子数组 I 思路: 取 nums[1:] 的最小2个值 可以部分排序&#xff0c;这样更快捷 class Solution {public int minimumCost(in…

Github操作网络异常笔记

Github操作网络异常笔记 1. 源由2. 解决2.1 方案一2.2 方案二 3. 总结 1. 源由 开源技术在国内永远是“蛋疼”&#xff0c;这些"政治"问题对于追求技术的我们&#xff0c;形成无法回避的障碍。 $ git pull ssh: connect to host github.com port 22: Connection ti…

Qt QCustomPlot 绘制子轴

抄大神杰作&#xff1a;QCustomplot&#xff08;五&#xff09;QCPAxisRect进行子绘图-CSDN博客文章浏览阅读5.9k次&#xff0c;点赞7次&#xff0c;收藏60次。文中介绍了QCustomPlot 子绘图需要掌握的类&#xff0c;也就是Matlab中的subplot&#xff0c;最后给出了一个完整的例…

EtherNet/IP开发:C++搭建基础模块,EtherNet/IP源代码

这里是CIP资料的协议层级图&#xff0c;讲解协议构造。 ODVA&#xff08;www.ODVA.org&#xff09;成立于1995年&#xff0c;是一个全球性协会&#xff0c;其成员包括世界领先的自动化公司。结合其成员的支持&#xff0c;ODVA的使命是在工业自动化中推进开放、可互操作的信息和…

人工智能原理实验1(2)——传教士与野人问题

&#x1f9e1;&#x1f9e1;实验内容&#x1f9e1;&#x1f9e1; 有n个牧师和n个野人准备渡河&#xff0c;但只有一条能容纳c个人的小船&#xff0c;为了防止野人侵犯牧师&#xff0c;要求无论在何处&#xff0c;牧师的人数不得少于野人的人数(除非牧师人数为0)&#xff0c;且假…

算法练习-替换数字(思路+流程图+代码)

难度参考 难度&#xff1a;简单 分类&#xff1a;字符串 难度与分类由我所参与的培训课程提供&#xff0c;但需要注意的是&#xff0c;难度与分类仅供参考。以下内容均为个人笔记&#xff0c;旨在督促自己认真学习。 题目 给定一个字符串S,它包含小写字母和数字字符&#xff0…

全国各省市上市公司数量数据,Shp、excel格式,含上市企业数量、行政区划中心点位经纬度等字段

基本信息. 数据名称: 全国各省市上市公司数量数据 数据格式: Shp、excel 数据时间: 2023年1月 数据几何类型: 面 数据坐标系: WGS84 数据来源&#xff1a;网络公开数据 数据字段&#xff1a; 序号字段名称字段说明1province省份名称2provin_dm省份代码3city城市名…

力扣刷MySQL-第七弹(详细讲解)

&#x1f389;欢迎您来到我的MySQL基础复习专栏 ☆* o(≧▽≦)o *☆哈喽~我是小小恶斯法克&#x1f379; ✨博客主页&#xff1a;小小恶斯法克的博客 &#x1f388;该系列文章专栏&#xff1a;力扣刷题讲解-MySQL &#x1f379;文章作者技术和水平很有限&#xff0c;如果文中出…

如何快速搭建springboot+前后端分离(vue),多商户客户端实现微信小程序+ios+app使用uniapp(一处编写,处处编译)

kxmalls外卖生鲜多商户&#xff0c;针对中小商户、企业和个人学习者开发。使用Java编码&#xff0c;采用SpringBoot、Mybatis-Plus等易用框架&#xff0c;适合个人学习研究。同时支持单机部署、集群部署&#xff0c;用户与店铺范围动态定位&#xff0c;中小商户企业可根据业务动…

C语言的编译和链接

每日一言 要保持希望在每天清晨太阳升起。 --自己 前言 当我们写下C语言代码&#xff08;源文件、以.c为后缀&#xff09;的时候&#xff0c;他需要经过一个翻译环境&#xff0c;被处理后形成一个可执行程序&#xff08;以.exe为后缀&#xff09;。形成的这个可执行程序里面放…

【服务器】搭建一台属于自己的服务器

​🌈个人主页:Sarapines Programmer🔥 系列专栏:【服务器】搭建网站⏰诗赋清音:云生高巅梦远游, 星光点缀碧海愁。 山川深邃情难晤, 剑气凌云志自修。 目录 1. 购买服务器和域名 1.1 购买服务器 1.1.1 阿里云服务器 1.1.2 香草云服务器 1.2 购买域名 2. 安装宝塔…

matlab抽取与插值

什么是抽取&#xff1f; 我们假设一个数字信号 x ( n ) , n 1 , 2 , . . . , N x(n),n1,2,...,N x(n),n1,2,...,N共有 N N N个点&#xff0c;抽取就是每个几个点抽1个点&#xff0c;比如2倍抽取&#xff0c;那么抽取后的信号为 y ( n ) , y ( 1 ) x ( 1 ) , y ( 2 ) x ( 3 …

WebSocket-黑马好客租房

文章目录 网站中的消息功能如何实现&#xff1f;什么是WebSocket&#xff1f;http与websocket的区别httpwebsocket 浏览器支持情况快速入门创建itcast-websocket工程websocket的相关注解说明实现websocket服务测试编写js客户端 SpringBoot整合WebSocket导入依赖编写WebSocketHa…

Python实现M-Estimators稳健线性回归模型(RLM算法)项目实战

说明&#xff1a;这是一个机器学习实战项目&#xff08;附带数据代码文档视频讲解&#xff09;&#xff0c;如需数据代码文档视频讲解可以直接到文章最后获取。 1.项目背景 M-Estimators 是稳健统计估计中的一个重要概念&#xff0c;它们在处理含有异常值、离群点或者影响点的…

GPT应用_AutoGPT

项目地址&#xff1a;https://github.com/Significant-Gravitas/AutoGPT 1 功能 1.1 整体功能&#xff0c;想解决什么问题 单独使用 ChatGPT 时&#xff0c;只提供基本的聊天&#xff0c;无法实现复杂多步的功能&#xff0c;以及与其它应用交互&#xff0c;如果想提供某种功…

【开源】基于JAVA的停车场收费系统

目录 一、摘要1.1 项目介绍1.2 项目录屏 二、功能模块2.1 停车位模块2.2 车辆模块2.3 停车收费模块2.4 IC卡模块2.5 IC卡挂失模块 三、系统设计3.1 用例设计3.2 数据库设计3.2.1 停车场表3.2.2 车辆表3.2.3 停车收费表3.2.4 IC 卡表3.2.5 IC 卡挂失表 四、系统实现五、核心代码…