设计模式——模板模式
定义与基本概念
- 模板模式(Template Pattern)是一种行为设计模式。它在一个抽象类中定义了一个操作的算法骨架,将一些步骤的实现延迟到具体子类中。这个抽象类就像是一个模板,定义了执行某个流程的基本框架,而具体的细节可以由不同的子类根据自身的需求来填充。
- 例如,在制作饮品的过程中,制作饮品的流程(如准备材料、混合材料、添加调料、包装等)可以看作是一个模板。不同的饮品(如咖啡、茶)在这个流程的某些具体步骤(如混合材料、添加调料)上会有不同的做法,但整体的流程框架是相似的。
结构组成
- 抽象模板(Abstract Template)类:
它定义了一个或多个抽象方法,这些方法代表了算法中的某些步骤,其具体实现将由具体子类完成。同时,抽象模板类还定义了一个模板方法(Template Method),这个方法规定了算法的骨架,它按照一定的顺序调用其他方法(包括抽象方法和具体方法)。以饮品制作的例子来说,抽象模板类可能定义了prepare()(抽象方法,准备材料)、mix()(抽象方法,混合材料)、addSeasoning()(抽象方法,添加调料)、package()(具体方法,包装)和makeDrink()(模板方法)。makeDrink()方法的实现可能是按照准备材料、混合材料、添加调料、包装的顺序调用其他方法。 - 具体模板(Concrete Template)类:
继承自抽象模板类,实现了抽象模板类中的抽象方法。每个具体模板类代表了一种具体的实现方式,根据不同的业务需求来填充抽象方法的具体内容。在饮品制作的例子中,Coffee类和Tea类是具体模板类,Coffee类的prepare()方法可能是准备咖啡豆和水,mix()方法可能是研磨咖啡豆后煮咖啡,addSeasoning()方法可能是加糖和奶;而Tea类的prepare()方法可能是准备茶叶和水,mix()方法可能是泡茶,addSeasoning()方法可能是加柠檬片。
工作原理
- 客户端代码创建具体模板类的对象,然后调用抽象模板类中定义的模板方法。模板方法在执行过程中,会按照预先定义的顺序调用其他方法,其中抽象方法的具体实现由具体模板类提供。这样,不同的具体模板类就可以在遵循相同算法骨架的基础上,实现不同的具体行为。
- 例如,在一个数据处理系统中,有一个抽象的数据处理模板类,它定义了loadData()(抽象方法)、processData()(抽象方法)、saveData()(具体方法)和execute()(模板方法)。execute()方法按照先loadData(),再processData(),最后saveData()的顺序调用其他方法。具体的数据处理子类(如TextDataProcessor和ImageDataProcessor)实现了loadData()和processData()方法,以适应不同类型数据的处理方式。当客户端代码调用TextDataProcessor对象的execute()方法时,就会按照TextDataProcessor类实现的loadData()和processData()方法以及DataProcessor类(抽象模板类)定义的saveData()方法来完成文本数据的处理流程。
代码示例
以下是一个简单的 C++ 模板模式示例,以文件处理为例。
- 首先是抽象模板类:
class FileHandler {
public:void processFile() {openFile();readFile();processContents();closeFile();}virtual void openFile() = 0;virtual void readFile() = 0;virtual void processContents() = 0;void closeFile() {std::cout << "文件已关闭。" << std::endl;}
};
- 然后是具体模板类,以文本文件处理为例:
class TextFileHandler : public FileHandler {
public:void openFile() override {std::cout << "打开文本文件。" << std::endl;}void readFile() override {std::cout << "读取文本文件内容。" << std::endl;}void processContents() override {std::cout << "处理文本文件内容。" << std::endl;}
};
- 另一个具体模板类,以二进制文件处理为例:
class BinaryFileHandler : public FileHandler {
public:void openFile() override {std::cout << "打开二进制文件。" << std::endl;}void readFile() override {std::cout << "读取二进制文件内容。" << std::endl;}void processContents() override {std::cout << "处理二进制文件内容。" << std::endl;}
};
- 使用示例:
int main() {TextFileHandler textHandler;textHandler.processFile();std::cout << "------------------------" << std::endl;BinaryFileHandler binaryHandler;binaryHandler.processFile();return 0;
}
优点
- 代码复用性高:
抽象模板类定义的算法骨架可以被多个具体子类复用。在上述文件处理的例子中,processFile()方法的流程(打开文件、读取文件、处理内容、关闭文件)在文本文件处理和二进制文件处理中都可以使用,减少了代码的重复编写。 - 可维护性好:
由于算法的骨架和具体步骤的实现分离,当需要修改算法的整体流程时,只需要在抽象模板类中修改模板方法;当需要修改某个具体步骤的实现时,只需要在相应的具体子类中进行修改。例如,在一个软件系统的用户注册流程中,如果需要修改注册的整体流程(如增加验证步骤),可以在抽象的注册模板类中修改模板方法;如果需要修改某个验证方式(如密码验证),可以在具体的注册子类中修改相应的抽象方法的实现。 - 符合开闭原则:
可以很容易地通过创建新的具体子类来扩展系统的功能,而不需要修改已有的代码。比如,在文件处理系统中,如果要增加一种新的文件类型(如 XML 文件)的处理,只需要创建一个新的XMLFileHandler类,实现抽象模板类中的抽象方法即可。
缺点
- 可能导致类层次复杂:
如果有多个抽象方法需要具体子类实现,并且具体子类的数量较多,会导致类层次结构变得复杂。例如,在一个复杂的工作流系统中,每个工作流步骤都可能有多个抽象方法需要具体子类实现,随着工作流类型的增加,类的数量和层次结构会变得难以管理。 - 子类的灵活性有限:
具体子类必须遵循抽象模板类定义的算法骨架,这在一定程度上限制了子类的灵活性。如果某个具体子类需要对算法骨架进行较大的修改,可能需要重新设计整个类结构,或者采用其他设计模式来解决。
相关文章:
设计模式——模板模式
定义与基本概念 模板模式(Template Pattern)是一种行为设计模式。它在一个抽象类中定义了一个操作的算法骨架,将一些步骤的实现延迟到具体子类中。这个抽象类就像是一个模板,定义了执行某个流程的基本框架,而具体的细…...
CV22_语义分割基础
1. 常见的分割类型 在计算机视觉领域,根据不同的应用场景和需求,分割任务可以分为几种主要类型。以下是几种常见的分割类型: 语义分割(Semantic Segmentation): 语义分割的目标是将图像中的每个像素分配到…...
Dubbo源码解析-Dubbo的线程模型(九)
一、Dubbo线程模型 首先明确一个基本概念:IO 线程和业务线程的区别 IO 线程:配置在netty 连接点的用于处理网络数据的线程,主要处理编解码等直接与网络数据 打交道的事件。 业务线程:用于处理具体业务逻辑的线程,可以…...
【Canvas与标志】圆角三角形生化危险警示标志
【成图】 【代码】 <!DOCTYPE html> <html lang"utf-8"> <meta http-equiv"Content-Type" content"text/html; charsetutf-8"/> <head><title>圆角三角形生化危险警示标志 Draft1</title><style type&qu…...
解决Dcat Admin laravel框架登录报错问题,(blocked:mixed-content)
前言 在使用 Dcat Admin 后台登录时,发生 error 报错:(blocked:mixed-content) xhr VM484:1,浏览器拦截 其实这是浏览器在 HTTPS 页面中尝试加载 HTTP 资源,导致浏览器阻止了这些不安全的请求。 解决 在 .env 文件中添加或修改 AD…...
(三)Sping Boot学习——升级jdk1.8-jdk18
1.修改系统环境变量。 2.idea中修改配置。 3.项目setting中设置修改 4.更新后还要重新下载依赖mvn clean install ,并且记住reload 项目。同时查看java -version查看一下jdk版本。...
语言模型中的多模态链式推理
神经网络的公式推导 简介摘要引言多模态思维链推理的挑战多模态CoT框架多模态CoT模型架构细节编码模块融合模块解码模块 实验结果运行代码补充细节安装包下载Flan-T5数据集准备rougenltkall-MiniLM-L6-v2运行 简介 本文主要对2023一篇论文《Multimodal Chain-of-Thought Reason…...
SCons:下一代构建工具,如何用 Python 驱动高效构建?
在现代软件开发中,构建工具是开发流程中不可或缺的一环。无论是小型项目还是跨平台的复杂工程,选择一个高效、灵活的工具都能显著提高开发效率和代码质量。SCons,一个以 Python 为基础的构建工具,通过自动化依赖管理、灵活的扩展性…...
springboot 整合 rabbitMQ (延迟队列)
前言: 延迟队列是一个内部有序的数据结构,其主要功能体现在其延时特性上。这种队列存储的元素都设定了特定的处理时间,意味着它们需要在规定的时间点或者延迟之后才能被取出并进行相应的处理。简而言之,延时队列被设计用于存放那…...
ES 基本使用与二次封装
概述 基本了解 Elasticsearch 是一个开源的分布式搜索和分析引擎,基于 Apache Lucene 构建。它提供了对海量数据的快速全文搜索、结构化搜索和分析功能,是目前流行的大数据处理工具之一。主要特点即高效搜索、分布式存储、拓展性强 核心功能 全文搜索:…...
分割一切2.0,SAM2详解
🏡作者主页:点击! 🤖编程探索专栏:点击! ⏰️创作时间:2024年11月24日20点03分 神秘男子影, 秘而不宣藏。 泣意深不见, 男子自持重, 子夜独自沉。 论文链接 点击开启你的论文编程之旅…...
Spring AI Fluent API:与AI模型通信的流畅体验
引言 随着人工智能(AI)技术的飞速发展,越来越多的应用场景开始融入AI技术以提升用户体验和系统效率。在Java开发中,与AI模型通信成为了一个重要而常见的需求。为了满足这一需求,Spring AI引入了ChatClient,…...
基于python的长津湖评论数据分析与可视化,使用是svm情感分析建模
引言 研究背景及意义 上世纪初开始,中国电影就以自己独有的姿态登上了世界电影史的舞台。中国电影作为国家文化和思想观念的反映与延伸,能够增强文化自信,在文化输出方面有着极其重要的作用1[1]。 改革开放以来,随着生产力的提高…...
Lucene(2):Springboot整合全文检索引擎TermInSetQuery应用实例附源码
前言 本章代码已分享至Gitee: https://gitee.com/lengcz/springbootlucene01 接上文。Lucene(1):Springboot整合全文检索引擎Lucene常规入门附源码 如何在指定范围内查询。从lucene 7 开始,filter 被弃用,导致无法进行调节过滤。 TermInSetQuery 指定…...
shell完结
声明! 学习视频来自B站up主 **泷羽sec** 有兴趣的师傅可以关注一下,如涉及侵权马上删除文章,笔记只是方便各位师傅的学习和探讨,文章所提到的网站以及内容,只做学习交流,其他均与本人以及泷羽sec团队无关&a…...
【2024最新】基于Springboot+Vue的智慧食堂系统Lw+PPT
作者:计算机搬砖家 开发技术:SpringBoot、php、Python、小程序、SSM、Vue、MySQL、JSP、ElementUI等,“文末源码”。 专栏推荐:SpringBoot项目源码、Vue项目源码、SSM项目源码、微信小程序源码 精品专栏:Java精选实战项…...
NVR小程序接入平台EasyNVR多品牌NVR管理工具:高效管理分散视频资源的解决方案
在当今数字化、智能化的时代背景下,视频监控已成为各行各业不可或缺的一部分,从公共安全到企业运维,再到智慧城市建设,视频资源的管理与应用正面临着前所未有的挑战。如何高效整合、管理这些遍布各地的分散视频资源,成…...
排序算法(三)--插入排序
文章目录 一、插入排序的基本原理二、插入排序的C语言实现三、代码解析 插入排序 C语言实例 一、插入排序的基本原理 插入排序的基本思想是将数组中的元素逐一取出,然后将其插入到已经排好序的部分中的适当位置,直到整个数组排序完成。具体步骤如下&…...
YOLOv11融合[ECCV 2018]RCAN中的RCAB模块及相关改进思路
YOLOv11v10v8使用教程: YOLOv11入门到入土使用教程 YOLOv11改进汇总贴:YOLOv11及自研模型更新汇总 《Image Super-Resolution Using Very Deep Residual Channel Attention Networks》 一、 模块介绍 论文链接:https://arxiv.org/abs/1807…...
排序(Java数据结构)
1. 排序的概念及引用 1.1 排序的概念 排序:所谓排序,就是使一串记录,按照其中的某个或某些关键字的大小,递增或递减的排列起来的操作。(所有的排序都是默认从小到大排序) 稳定性:假定在待排序的记录序列中ÿ…...
未来机器人的大脑:如何用神经网络模拟器实现更智能的决策?
编辑:陈萍萍的公主一点人工一点智能 未来机器人的大脑:如何用神经网络模拟器实现更智能的决策?RWM通过双自回归机制有效解决了复合误差、部分可观测性和随机动力学等关键挑战,在不依赖领域特定归纳偏见的条件下实现了卓越的预测准…...
linux之kylin系统nginx的安装
一、nginx的作用 1.可做高性能的web服务器 直接处理静态资源(HTML/CSS/图片等),响应速度远超传统服务器类似apache支持高并发连接 2.反向代理服务器 隐藏后端服务器IP地址,提高安全性 3.负载均衡服务器 支持多种策略分发流量…...
JavaScript 中的 ES|QL:利用 Apache Arrow 工具
作者:来自 Elastic Jeffrey Rengifo 学习如何将 ES|QL 与 JavaScript 的 Apache Arrow 客户端工具一起使用。 想获得 Elastic 认证吗?了解下一期 Elasticsearch Engineer 培训的时间吧! Elasticsearch 拥有众多新功能,助你为自己…...
蓝桥杯 2024 15届国赛 A组 儿童节快乐
P10576 [蓝桥杯 2024 国 A] 儿童节快乐 题目描述 五彩斑斓的气球在蓝天下悠然飘荡,轻快的音乐在耳边持续回荡,小朋友们手牵着手一同畅快欢笑。在这样一片安乐祥和的氛围下,六一来了。 今天是六一儿童节,小蓝老师为了让大家在节…...
Python实现prophet 理论及参数优化
文章目录 Prophet理论及模型参数介绍Python代码完整实现prophet 添加外部数据进行模型优化 之前初步学习prophet的时候,写过一篇简单实现,后期随着对该模型的深入研究,本次记录涉及到prophet 的公式以及参数调优,从公式可以更直观…...
ETLCloud可能遇到的问题有哪些?常见坑位解析
数据集成平台ETLCloud,主要用于支持数据的抽取(Extract)、转换(Transform)和加载(Load)过程。提供了一个简洁直观的界面,以便用户可以在不同的数据源之间轻松地进行数据迁移和转换。…...
Psychopy音频的使用
Psychopy音频的使用 本文主要解决以下问题: 指定音频引擎与设备;播放音频文件 本文所使用的环境: Python3.10 numpy2.2.6 psychopy2025.1.1 psychtoolbox3.0.19.14 一、音频配置 Psychopy文档链接为Sound - for audio playback — Psy…...
微服务商城-商品微服务
数据表 CREATE TABLE product (id bigint(20) UNSIGNED NOT NULL AUTO_INCREMENT COMMENT 商品id,cateid smallint(6) UNSIGNED NOT NULL DEFAULT 0 COMMENT 类别Id,name varchar(100) NOT NULL DEFAULT COMMENT 商品名称,subtitle varchar(200) NOT NULL DEFAULT COMMENT 商…...
管理学院权限管理系统开发总结
文章目录 🎓 管理学院权限管理系统开发总结 - 现代化Web应用实践之路📝 项目概述🏗️ 技术架构设计后端技术栈前端技术栈 💡 核心功能特性1. 用户管理模块2. 权限管理系统3. 统计报表功能4. 用户体验优化 🗄️ 数据库设…...
Vite中定义@软链接
在webpack中可以直接通过符号表示src路径,但是vite中默认不可以。 如何实现: vite中提供了resolve.alias:通过别名在指向一个具体的路径 在vite.config.js中 import { join } from pathexport default defineConfig({plugins: [vue()],//…...
