设计模式——模板模式
定义与基本概念
- 模板模式(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 排序的概念 排序:所谓排序,就是使一串记录,按照其中的某个或某些关键字的大小,递增或递减的排列起来的操作。(所有的排序都是默认从小到大排序) 稳定性:假定在待排序的记录序列中ÿ…...
eNSP-Cloud(实现本地电脑与eNSP内设备之间通信)
说明: 想象一下,你正在用eNSP搭建一个虚拟的网络世界,里面有虚拟的路由器、交换机、电脑(PC)等等。这些设备都在你的电脑里面“运行”,它们之间可以互相通信,就像一个封闭的小王国。 但是&#…...
Oracle查询表空间大小
1 查询数据库中所有的表空间以及表空间所占空间的大小 SELECTtablespace_name,sum( bytes ) / 1024 / 1024 FROMdba_data_files GROUP BYtablespace_name; 2 Oracle查询表空间大小及每个表所占空间的大小 SELECTtablespace_name,file_id,file_name,round( bytes / ( 1024 …...
【单片机期末】单片机系统设计
主要内容:系统状态机,系统时基,系统需求分析,系统构建,系统状态流图 一、题目要求 二、绘制系统状态流图 题目:根据上述描述绘制系统状态流图,注明状态转移条件及方向。 三、利用定时器产生时…...
从零实现STL哈希容器:unordered_map/unordered_set封装详解
本篇文章是对C学习的STL哈希容器自主实现部分的学习分享 希望也能为你带来些帮助~ 那咱们废话不多说,直接开始吧! 一、源码结构分析 1. SGISTL30实现剖析 // hash_set核心结构 template <class Value, class HashFcn, ...> class hash_set {ty…...
Android Bitmap治理全解析:从加载优化到泄漏防控的全生命周期管理
引言 Bitmap(位图)是Android应用内存占用的“头号杀手”。一张1080P(1920x1080)的图片以ARGB_8888格式加载时,内存占用高达8MB(192010804字节)。据统计,超过60%的应用OOM崩溃与Bitm…...
全面解析各类VPN技术:GRE、IPsec、L2TP、SSL与MPLS VPN对比
目录 引言 VPN技术概述 GRE VPN 3.1 GRE封装结构 3.2 GRE的应用场景 GRE over IPsec 4.1 GRE over IPsec封装结构 4.2 为什么使用GRE over IPsec? IPsec VPN 5.1 IPsec传输模式(Transport Mode) 5.2 IPsec隧道模式(Tunne…...
分布式增量爬虫实现方案
之前我们在讨论的是分布式爬虫如何实现增量爬取。增量爬虫的目标是只爬取新产生或发生变化的页面,避免重复抓取,以节省资源和时间。 在分布式环境下,增量爬虫的实现需要考虑多个爬虫节点之间的协调和去重。 另一种思路:将增量判…...
中医有效性探讨
文章目录 西医是如何发展到以生物化学为药理基础的现代医学?传统医学奠基期(远古 - 17 世纪)近代医学转型期(17 世纪 - 19 世纪末)现代医学成熟期(20世纪至今) 中医的源远流长和一脉相承远古至…...
【Linux系统】Linux环境变量:系统配置的隐形指挥官
。# Linux系列 文章目录 前言一、环境变量的概念二、常见的环境变量三、环境变量特点及其相关指令3.1 环境变量的全局性3.2、环境变量的生命周期 四、环境变量的组织方式五、C语言对环境变量的操作5.1 设置环境变量:setenv5.2 删除环境变量:unsetenv5.3 遍历所有环境…...
Axure 下拉框联动
实现选省、选完省之后选对应省份下的市区...
