设计模式-08 - 模板方法模式 Template Method
设计模式-08 - 模板方法模式 Template Method
1.定义
模板方法模式是一种设计模式,它定义了一个操作的骨架,而由子类来决定如何实现该操作的某些步骤。它允许子类在不改变算法结构的情况下重定义算法的特定步骤。
模板方法模式适合用于以下情况:
需要定义一个算法的骨架,但允许子类实现特定步骤:例如,一个排序算法,其中骨架定义了排序过程,而子类定义了具体的比较方法。
需要避免在子类中重复代码:例如,一个图形绘制框架,其中抽象类定义了绘制过程的骨架,而子类定义了绘制特定形状的具体方法。
需要提供算法的可扩展性和灵活性:例如,一个游戏引擎,其中抽象类定义了游戏循环的骨架,而子类定义了特定游戏玩法的具体步骤。
2.内涵
模板方法模式具有以下优点:
可扩展性:通过允许子类重定义某些步骤,模板方法模式提供了很大的可扩展性。
代码复用:模板方法模式允许在抽象类中定义算法的骨架,从而避免在子类中重复代码。
灵活性:子类可以根据需要自定义算法的特定步骤,从而提供灵活性。
模板方法模式也有一些缺点:
可能难以理解:模板方法模式可能难以理解,尤其是对于大型和复杂的算法。
难以调试:由于算法的骨架在抽象类中定义,因此可能难以调试子类中的特定步骤。
3.使用示例
#include <iostream>// Step 1: Template Method (Abstract Class) 定义模板
class VehicleTemplate {
public:// Template method defines the algorithm structurevoid buildVehicle() {assembleBody();installEngine();addWheels();std::cout << "Vehicle is ready!\n";}// Abstract methods to be implemented by concrete classesvirtual void assembleBody() = 0;virtual void installEngine() = 0;virtual void addWheels() = 0;
};// Step 2: Concrete Classes 具体实现类
class Car : public VehicleTemplate {
public:void assembleBody() override {std::cout << "Assembling car body.\n";}void installEngine() override {std::cout << "Installing car engine.\n";}void addWheels() override {std::cout << "Adding 4 wheels to the car.\n";}
};class Motorcycle : public VehicleTemplate {
public:void assembleBody() override {std::cout << "Assembling motorcycle frame.\n";}void installEngine() override {std::cout << "Installing motorcycle engine.\n";}void addWheels() override {std::cout << "Adding 2 wheels to the motorcycle.\n";}
};// Step 3: Client Code 客户端调用代码
int main() {std::cout << "Building a Car:\n";Car car;car.buildVehicle();std::cout << "\nBuilding a Motorcycle:\n";Motorcycle motorcycle;motorcycle.buildVehicle();return 0;
}
4.注意事项
在使用模板方法模式时,需要注意以下事项:
- 算法的灵活性:确保算法的骨架足够灵活,以允许子类根据需要自定义算法的特定步骤。
- 抽象和具体方法的平衡:抽象类应仅包含算法的骨架,而具体子类应包含算法的具体实现。避免在抽象类中包含具体细节,因为这会降低可扩展性。
- 继承层次结构:仔细考虑继承层次结构,并确保子类不会引入不必要的复杂性或冗余。
- 可测试性:确保抽象类和具体子类都可测试,以验证算法的正确性。
- 性能影响:模板方法模式可能引入一些性能开销,因为需要在运行时调用模板方法和具体方法。在对性能敏感的应用程序中,需要考虑这种开销。
此外,还有一些最佳实践可以提高模板方法模式的使用效果:
- 使用钩子方法:钩子方法是抽象类中可选的方法,允许子类在不重写整个模板方法的情况下自定义算法的特定方面。
- 提供默认实现:对于某些步骤,可以提供默认实现,以简化子类的实现。
- 使用多态性:利用多态性来允许子类在运行时替换算法的具体步骤。
- 考虑使用策略模式:在某些情况下,策略模式可以是模板方法模式的替代方案,它允许算法的变体完全由策略对象决定。
5.最佳实践
模板方法模式的最佳实践,避免错误的最佳经验:
- 明确定义算法的骨架:抽象类应明确定义算法的骨架,包括其步骤和顺序。避免在抽象类中包含具体细节。
- 只在具体子类中实现具体步骤:具体子类应仅实现算法的具体步骤,而不要改变算法的整体结构。
- 使用钩子方法:钩子方法允许子类在不重写整个模板方法的情况下自定义算法的特定方面。这可以提高灵活性并避免不必要的代码重复。
- 提供默认实现:对于某些步骤,可以提供默认实现,以简化子类的实现。默认实现应涵盖常见情况,而子类可以根据需要重写它们。
- 考虑使用策略模式:在某些情况下,策略模式可以是模板方法模式的替代方案,它允许算法的变体完全由策略对象决定。策略模式可以提供更大的灵活性,但可能更复杂。
- 确保可测试性:抽象类和具体子类都应可测试,以验证算法的正确性。使用单元测试可以确保算法在各种情况下都能按预期工作。
- 考虑性能影响:模板方法模式可能引入一些性能开销,因为需要在运行时调用模板方法和具体方法。在对性能敏感的应用程序中,需要考虑这种开销。
6.总结
模板方法模式适合用于以下情况:
- 需要定义一个算法的骨架,但允许子类实现特定步骤:例如,一个排序算法,其中骨架定义了排序过程,而子类定义了具体的比较方法。
- 需要避免在子类中重复代码:例如,一个图形绘制框架,其中抽象类定义了绘制过程的骨架,而子类定义了绘制特定形状的具体方法。
- 需要提供算法的可扩展性和灵活性:例如,一个游戏引擎,其中抽象类定义了游戏循环的骨架,而子类定义了特定游戏玩法的具体步骤。
总体而言,模板方法模式是一种强大的设计模式,可以提高代码的可扩展性、复用性和灵活性。通过仔细考虑上述注意事项和最佳实践,可以有效地使用该模式来设计和实现复杂的算法。
相关文章:
设计模式-08 - 模板方法模式 Template Method
设计模式-08 - 模板方法模式 Template Method 1.定义 模板方法模式是一种设计模式,它定义了一个操作的骨架,而由子类来决定如何实现该操作的某些步骤。它允许子类在不改变算法结构的情况下重定义算法的特定步骤。 模板方法模式适合用于以下情况&am…...
Android 适配阿拉伯语之vector图标镜像
Android 适配阿拉伯语之vector图标镜像 android:autoMirrored“true” 属性简单而直接的方法来自动处理 RTL 环境中图标的翻转。 使用 android:autoMirrored“true” 在 Vector Drawable 中是一种非常方便的方法,因为它允许你使用相同的 drawable 资源来适应不同的…...
推荐4个可用的github国内镜像
Github是全球最大的代码托管云平台,超过1亿用户在平台上分享代码及数据,深受生物信息学软件开发者的喜爱,并且现在发表文章,若涉及到代码,编辑还要求我们把代码及数据存放在github上,以便检查数据的真实性和…...
从项目开始学习Vue——02(若依框架)
往期: 从项目开始学习Vue——01 目录标题 一、基础插件(一)路由Vue Router(二)导航守卫(路由拦截器)二、Vuex(一)什么是VuexVuex的部分介绍内容: (…...
使用JavaScript日历小部件和DHTMLX Gantt的应用场景(二)
DHTMLX Suite UI 组件库允许您更快地构建跨平台、跨浏览器 Web 和移动应用程序。它包括一组丰富的即用式 HTML5 组件,这些组件可以轻松组合到单个应用程序界面中。 DHTMLX Gantt是用于跨浏览器和跨平台应用程序的功能齐全的Gantt图表,可满足项目管理应用…...
springboot整合redis多数据源(附带RedisUtil)
单数据源RedisUtil(静态) 单数据源RedisUtil,我这里implements ApplicationContextAware在setApplicationContext注入redisTemplate,工具类可以直接类RedisUtil.StringOps.get()使用 package com.vehicle.manager.core.util;import com.alibaba.fastjson.JSON; import lombok.e…...
Web实时通信的学习之旅:SSE(Server-Sent Events)的技术详解及简单示例演示
文章目录 一、什么是SSE二、SSE技术的基本原理三、SSE适用于场景四、Node服务端示例1、协议2、格式3、事件3.1、事件3.2、事件唯一标识符3.3、重连事件 4、具体示例 五、客户端示例1、检测客户端是否支持SSE2、创建客户端连接3、事件监听4、接收事件5、自定义事件6、错误处理7、…...
Apache Flume事务
Apache Flume 中的事务处理是指 Flume Agent 在处理事件流时的一种机制,用于确保数据的可靠传输和处理。 1. 事务概述: Flume 中的事务是指一组事件的传输和处理,这些事件在传输过程中要么全部成功完成,要么全部失败࿰…...
根据部门id删除该部门下的员工(事务)
application.properties: 或: application.yml: 新表: 日志对象类: 日志service类: 日志service接口: 日志mapper类: 部门service类: 员工mapper类:...
Java之String类
一、String类常用方法 1.引用类型的比较 我们知道在Java中两个引用遍历是不能用" "号来比较的,而String类重写了父类objects的equals方法, 实现了引用类型的比较 例子 import java.util.Scanner; public class Main { public static void…...
es终止快照恢复进程的方法
方法1、删除索引可以终止,恢复进程。 DELETE index_* // 按通配符删除以index_开头的索引 DELETE _all // 删除全部索引 POST *,-.*/_close 关闭索引 POST *,-.*/_open 打开索引 DELETE *,-.* 删除全部索引方法2、强制重启es 集群也可也终…...
ubantu安装rabbbitmq
ubantu安装rabbbitmq 安装Erlang1、在linux下直接安装2、上传Erlang文件后解压 安装rabbitmq开启web管理接口创建用户及修改guest密码,删除guest默认账号 安装Erlang 1、在linux下直接安装 #运行以下命令直接安装: sudo apt-get install erlang#可运行…...
了解 条码工具 Dynamsoft 在条码读取器中的形态运算
在图像处理中,术语形态学是指分析形状以填充小孔、去除噪声、提取轮廓等的一组操作。形态学操作很像空间卷积中的过滤过程。有两个部分在起作用:结构元素和预定义的计算规则。 点击下载Dynamsoft最新版https://www.evget.com/product/3691/download 结…...
NIO和NIO.2对比
Java NIO (New Input/Output) 是从Java 1.4版本开始引入的一个新的I/O API,用于替代原来的BIO(Blocking I/O)API。NIO提供了更加灵活和高效的网络通信方式,特别适合于高吞吐量的网络编程。NIO的主要特点是非阻塞模式,它…...
Google准备好了吗?OpenAI发布ChatGPT驱动搜索引擎|TodayAI
在科技界波澜壮阔的发展中,OpenAI正式宣布其最新突破——一个全新的基于ChatGPT技术的搜索引擎,旨在直接挑战谷歌在搜索领域的统治地位。这一创新将可能彻底改变用户上网搜索的方式。 据悉,这款AI驱动的搜索引擎利用了ChatGPT的强大功能&…...
乐观锁、悲观锁、互斥锁、读写锁
乐观锁和悲观锁是两种不同的锁机制,用于在多线程环境下解决资源竞争问题。互斥锁和读写锁是两种常见的锁类型,它们都可以用来实现乐观锁或悲观锁。 乐观锁 是一种无锁机制,它假设在多线程环境下对共享资源的操作不会发生冲突,因…...
Gin入门指南:从零开始快速掌握Go Web框架Gin
官网:https://gin-gonic.com/ GitHub:https://github.com/gin-gonic 了解 Gin Gin 是一个使用 Go 语言开发的 Web 框架,它非常轻量级且具有高性能。Gin 提供了快速构建 Web 应用程序所需的基本功能和丰富的中间件支持。 以下是 Gin 框架的一些特点和功能: 快速而高效:…...
Scala里的class、object、case class、case object 、trait
Class(类) 定义和作用 Scala 中的 class 是一种蓝图,用于创建对象(实例)。它定义了对象的状态和行为。类可以包含字段(属性)和方法(函数)。类可以有构造器,…...
移动端自动化测试工具 Appium 之 main 启动
文章目录 一、背景二、生成xml文件2.1、创建xml方法2.2、执行主类MainTest2.3、自动生成的xml2.4、工程目录2.5、执行结果 三、命令行执行appium服务四、主方法启动类五、集成Jenkins六、总结 一、背景 Jenkins 做集成测试是不错的工具,那么UI自动化是否可以&#…...
WT32-ETH01作为TCP Client进行通讯
目录 模块简介WT32-ETH01作为TCP Client设置电脑作为TCP Server设置连接并进行通讯总结 模块简介 WT32-ETH01网关主要功能特点: 采用双核Xtensa⑧32-bit LX6 MCU.集成SPI flash 32Mbit\ SRAM 520KB 支持TCP Server. TCP Client, UDP Server. UDP Client工作模式 支持串口、wif…...
Python爬虫实战:研究MechanicalSoup库相关技术
一、MechanicalSoup 库概述 1.1 库简介 MechanicalSoup 是一个 Python 库,专为自动化交互网站而设计。它结合了 requests 的 HTTP 请求能力和 BeautifulSoup 的 HTML 解析能力,提供了直观的 API,让我们可以像人类用户一样浏览网页、填写表单和提交请求。 1.2 主要功能特点…...
Linux链表操作全解析
Linux C语言链表深度解析与实战技巧 一、链表基础概念与内核链表优势1.1 为什么使用链表?1.2 Linux 内核链表与用户态链表的区别 二、内核链表结构与宏解析常用宏/函数 三、内核链表的优点四、用户态链表示例五、双向循环链表在内核中的实现优势5.1 插入效率5.2 安全…...
RocketMQ延迟消息机制
两种延迟消息 RocketMQ中提供了两种延迟消息机制 指定固定的延迟级别 通过在Message中设定一个MessageDelayLevel参数,对应18个预设的延迟级别指定时间点的延迟级别 通过在Message中设定一个DeliverTimeMS指定一个Long类型表示的具体时间点。到了时间点后…...
盘古信息PCB行业解决方案:以全域场景重构,激活智造新未来
一、破局:PCB行业的时代之问 在数字经济蓬勃发展的浪潮中,PCB(印制电路板)作为 “电子产品之母”,其重要性愈发凸显。随着 5G、人工智能等新兴技术的加速渗透,PCB行业面临着前所未有的挑战与机遇。产品迭代…...
从WWDC看苹果产品发展的规律
WWDC 是苹果公司一年一度面向全球开发者的盛会,其主题演讲展现了苹果在产品设计、技术路线、用户体验和生态系统构建上的核心理念与演进脉络。我们借助 ChatGPT Deep Research 工具,对过去十年 WWDC 主题演讲内容进行了系统化分析,形成了这份…...
R语言AI模型部署方案:精准离线运行详解
R语言AI模型部署方案:精准离线运行详解 一、项目概述 本文将构建一个完整的R语言AI部署解决方案,实现鸢尾花分类模型的训练、保存、离线部署和预测功能。核心特点: 100%离线运行能力自包含环境依赖生产级错误处理跨平台兼容性模型版本管理# 文件结构说明 Iris_AI_Deployme…...
高频面试之3Zookeeper
高频面试之3Zookeeper 文章目录 高频面试之3Zookeeper3.1 常用命令3.2 选举机制3.3 Zookeeper符合法则中哪两个?3.4 Zookeeper脑裂3.5 Zookeeper用来干嘛了 3.1 常用命令 ls、get、create、delete、deleteall3.2 选举机制 半数机制(过半机制࿰…...
条件运算符
C中的三目运算符(也称条件运算符,英文:ternary operator)是一种简洁的条件选择语句,语法如下: 条件表达式 ? 表达式1 : 表达式2• 如果“条件表达式”为true,则整个表达式的结果为“表达式1”…...
【git】把本地更改提交远程新分支feature_g
创建并切换新分支 git checkout -b feature_g 添加并提交更改 git add . git commit -m “实现图片上传功能” 推送到远程 git push -u origin feature_g...
Spring Cloud Gateway 中自定义验证码接口返回 404 的排查与解决
Spring Cloud Gateway 中自定义验证码接口返回 404 的排查与解决 问题背景 在一个基于 Spring Cloud Gateway WebFlux 构建的微服务项目中,新增了一个本地验证码接口 /code,使用函数式路由(RouterFunction)和 Hutool 的 Circle…...
