设计模式-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…...
Godep历史意义揭秘:Go依赖管理工具的开创者如何改变开发方式
Godep历史意义揭秘:Go依赖管理工具的开创者如何改变开发方式 【免费下载链接】godep dependency tool for go 项目地址: https://gitcode.com/gh_mirrors/go/godep Godep作为Go语言依赖管理工具的开创者,在Go生态系统的演进历程中扮演了至关重要的…...
CH340/CH341安卓USB主机模式开发实战
1. CH340/CH341安卓USB主机模式开发入门 很多开发者第一次接触安卓USB主机模式开发时,都会遇到一个典型问题:为什么我的手机连上CH340模块后毫无反应?这通常是因为安卓设备默认工作在从机模式(USB Device Mode),而连接串口设备需要…...
2021必修 首门CSS架构系统精讲 理论+实战玩转蘑菇街 百度网盘
在前端开发的职场鄙视链里,存在一个极其普遍的误区:认为电商页面就是“简单的列表详情”,没什么技术含量。殊不知,电商是前端技术最残酷的练兵场:毫秒级的首屏速度、像素级的视觉还原、千人千面的动态布局、以及大促期…...
像素风AI工具体验:像素史诗智识终端,让研究变得有趣又高效
像素风AI工具体验:像素史诗智识终端,让研究变得有趣又高效 1. 引言:当科研遇上像素冒险 想象一下:你是一位勇者,站在像素风格的城堡前,准备开始一场史诗般的冒险。但这次,你的武器不是剑与盾&…...
GLM-Image技术验证:长宽比对构图影响实测数据
GLM-Image技术验证:长宽比对构图影响实测数据 1. 项目背景介绍 GLM-Image是由智谱AI开发的先进文本到图像生成模型,提供了一个美观易用的Web交互界面。这个界面基于Gradio构建,让用户能够轻松使用GLM-Image模型生成高质量的AI图像。 在实际…...
Heygem数字人系统效果展示:看一段音频如何驱动多个数字人视频
Heygem数字人系统效果展示:看一段音频如何驱动多个数字人视频 1. 系统核心能力概览 Heygem数字人视频生成系统批量版webui版是一款基于AI技术的创新工具,能够将单一音频源同步驱动多个数字人视频生成。系统采用先进的语音驱动口型同步技术,…...
强化学习实战:Sarsa vs Q-learning,on-policy和off-policy到底怎么选?
强化学习实战:Sarsa与Q-learning的深度对比与策略选择指南 1. 理解策略分类的核心逻辑 在强化学习领域,策略选择直接影响算法的行为模式和学习效果。我们先从最基础的概念切入:什么是策略?简单来说,策略就是智能体在特…...
Phi-3-Mini-128K高性能推理优化:深入理解WSL2下的GPU资源调配
Phi-3-Mini-128K高性能推理优化:深入理解WSL2下的GPU资源调配 1. 引言 如果你是一位在Windows上搞AI开发的伙伴,可能早就受够了原生环境里那些烦人的依赖冲突和性能瓶颈。我也是这么过来的,直到开始用WSL2,感觉像是打开了新世界…...
从卡顿到流畅:Win11Debloat开源工具3步解决Windows系统优化难题
从卡顿到流畅:Win11Debloat开源工具3步解决Windows系统优化难题 【免费下载链接】Win11Debloat 一个简单的PowerShell脚本,用于从Windows中移除预装的无用软件,禁用遥测,从Windows搜索中移除Bing,以及执行各种其他更改…...
AI如何悄悄改变你的日常生活?5个你已离不开的AI应用场景
AI如何悄悄改变你的日常生活?5个你已离不开的AI应用场景 清晨被智能闹钟以最舒适的渐强音量唤醒,通勤路上听着音乐App精准推荐的歌单,晚上回家对着冰箱说出想吃的菜谱——这些场景中隐藏的AI技术,早已像水电一样成为生活基础设施。…...
