设计模式-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…...
应用升级/灾备测试时使用guarantee 闪回点迅速回退
1.场景 应用要升级,当升级失败时,数据库回退到升级前. 要测试系统,测试完成后,数据库要回退到测试前。 相对于RMAN恢复需要很长时间, 数据库闪回只需要几分钟。 2.技术实现 数据库设置 2个db_recovery参数 创建guarantee闪回点,不需要开启数据库闪回。…...

【JVM】- 内存结构
引言 JVM:Java Virtual Machine 定义:Java虚拟机,Java二进制字节码的运行环境好处: 一次编写,到处运行自动内存管理,垃圾回收的功能数组下标越界检查(会抛异常,不会覆盖到其他代码…...

关于iview组件中使用 table , 绑定序号分页后序号从1开始的解决方案
问题描述:iview使用table 中type: "index",分页之后 ,索引还是从1开始,试过绑定后台返回数据的id, 这种方法可行,就是后台返回数据的每个页面id都不完全是按照从1开始的升序,因此百度了下,找到了…...
基于Uniapp开发HarmonyOS 5.0旅游应用技术实践
一、技术选型背景 1.跨平台优势 Uniapp采用Vue.js框架,支持"一次开发,多端部署",可同步生成HarmonyOS、iOS、Android等多平台应用。 2.鸿蒙特性融合 HarmonyOS 5.0的分布式能力与原子化服务,为旅游应用带来…...

srs linux
下载编译运行 git clone https:///ossrs/srs.git ./configure --h265on make 编译完成后即可启动SRS # 启动 ./objs/srs -c conf/srs.conf # 查看日志 tail -n 30 -f ./objs/srs.log 开放端口 默认RTMP接收推流端口是1935,SRS管理页面端口是8080,可…...
Neo4j 集群管理:原理、技术与最佳实践深度解析
Neo4j 的集群技术是其企业级高可用性、可扩展性和容错能力的核心。通过深入分析官方文档,本文将系统阐述其集群管理的核心原理、关键技术、实用技巧和行业最佳实践。 Neo4j 的 Causal Clustering 架构提供了一个强大而灵活的基石,用于构建高可用、可扩展且一致的图数据库服务…...
三体问题详解
从物理学角度,三体问题之所以不稳定,是因为三个天体在万有引力作用下相互作用,形成一个非线性耦合系统。我们可以从牛顿经典力学出发,列出具体的运动方程,并说明为何这个系统本质上是混沌的,无法得到一般解…...
【碎碎念】宝可梦 Mesh GO : 基于MESH网络的口袋妖怪 宝可梦GO游戏自组网系统
目录 游戏说明《宝可梦 Mesh GO》 —— 局域宝可梦探索Pokmon GO 类游戏核心理念应用场景Mesh 特性 宝可梦玩法融合设计游戏构想要素1. 地图探索(基于物理空间 广播范围)2. 野生宝可梦生成与广播3. 对战系统4. 道具与通信5. 延伸玩法 安全性设计 技术选…...

学习STC51单片机32(芯片为STC89C52RCRC)OLED显示屏2
每日一言 今天的每一份坚持,都是在为未来积攒底气。 案例:OLED显示一个A 这边观察到一个点,怎么雪花了就是都是乱七八糟的占满了屏幕。。 解释 : 如果代码里信号切换太快(比如 SDA 刚变,SCL 立刻变&#…...

NXP S32K146 T-Box 携手 SD NAND(贴片式TF卡):驱动汽车智能革新的黄金组合
在汽车智能化的汹涌浪潮中,车辆不再仅仅是传统的交通工具,而是逐步演变为高度智能的移动终端。这一转变的核心支撑,来自于车内关键技术的深度融合与协同创新。车载远程信息处理盒(T-Box)方案:NXP S32K146 与…...