软件设计模式系列之二十四——模板方法模式

在软件设计领域,设计模式是一组被反复使用、多次实践验证的经典问题解决方案。其中,模板方法模式是一种行为型设计模式,用于定义一个算法的骨架,将算法中的某些步骤延迟到子类中实现,从而使子类可以重新定义算法的某些特定步骤,同时保持算法的整体结构不变。本文将深入探讨模板方法模式,包括其定义、举例、结构、实现步骤、代码实现、典型应用场景、优缺点、类似模式以及一个小结。

1 模式的定义

模板方法模式是一种行为型设计模式,它定义了一个算法的骨架,将算法的具体步骤延迟到子类中实现。这意味着,模板方法模式允许多个子类共享相同的算法框架,但可以在子类中实现各自特定的步骤,从而实现了代码的复用和扩展。通常,模板方法由一个抽象类定义,其中包含了算法的骨架,以及一些抽象方法,用于由子类来实现。

2 举例说明

在日常生活中,有许多例子符合模板方法模式的设计,这些例子都展示了一种在不同场景中重复使用相同的步骤或算法骨架的情况。以下是几个大家熟知的例子:

烹饪食物:烹饪是一个典型的模板方法模式示例。无论是制作披萨、汉堡、还是意大利面,都需要执行一系列共同的步骤,如准备食材、烹饪、装盘等。不同的食物有不同的特定步骤,可以在子类中实现。
在这里插入图片描述

咖啡店的饮料制作:就像前文提到的咖啡和茶的例子一样,咖啡店制作不同饮料也遵循相似的模板,包括烧水、冲泡、倒入杯子和加入调味品等步骤。

电子商务购物流程:在线购物网站通常具有相似的购物流程,包括浏览商品、将商品添加到购物车、填写配送信息、选择支付方式、确认订单等步骤。不同电子商务网站可以在这些步骤的执行顺序和细节上进行定制。

这些例子都展示了在日常生活中广泛存在的模板方法模式的应用,其中共享的步骤或算法骨架可以在不同的情境中重复使用,而特定的细节可以在子类或具体实例中定制。这种设计方法有助于提高效率、减少重复工作,并确保一致性。

3 结构

模板方法模式包含以下主要组成部分:
在这里插入图片描述

抽象类(Abstract Class):定义算法的骨架,包含一个或多个抽象方法,用于由子类实现。通常,抽象类还包含模板方法,该方法定义了算法的步骤顺序。

具体子类(Concrete Subclass):实现抽象类中的抽象方法,从而提供了算法的具体实现。

4 实现步骤

下面是使用模板方法模式的一般实现步骤:

创建一个抽象类,定义算法的骨架,并在其中声明抽象方法。

在抽象类中实现模板方法,该方法包含算法的步骤顺序,其中调用了抽象方法。

创建具体子类,继承自抽象类,并实现抽象方法以提供具体的算法实现。

在客户端代码中,使用具体子类来调用模板方法以执行算法。

5 代码实现

以下是一个使用Java语言实现的制作咖啡和茶的示例代码:

// 抽象类:饮料制备模板
abstract class Beverage {// 模板方法,制备饮料final void prepareBeverage() {boilWater();brew();pourInCup();addCondiments();}// 抽象方法:烧水abstract void boilWater();// 抽象方法:冲泡abstract void brew();// 抽象方法:倒入杯子abstract void pourInCup();// 抽象方法:加入调味品abstract void addCondiments();
}// 具体子类:咖啡
class Coffee extends Beverage {@Overridevoid boilWater() {System.out.println("烧水");}@Overridevoid brew() {System.out.println("冲泡咖啡");}@Overridevoid pourInCup() {System.out.println("倒入杯子");}@Overridevoid addCondiments() {System.out.println("加入糖和牛奶");}
}// 具体子类:茶
class Tea extends Beverage {@Overridevoid boilWater() {System.out.println("烧水");}@Overridevoid brew() {System.out.println("冲泡茶叶");}@Overridevoid pourInCup() {System.out.println("倒入杯子");}@Overridevoid addCondiments() {System.out.println("加入柠檬");}
}public class TemplateMethodPatternExample {public static void main(String[] args) {Beverage coffee = new Coffee();coffee.prepareBeverage();Beverage tea = new Tea();tea.prepareBeverage();}
}

6 典型应用场景

模板方法模式在实际软件开发中有许多应用场景,其中一些典型的包括:

框架设计:在框架中,模板方法模式用于定义框架的核心算法,而具体的功能由子类实现。例如,Java中的Servlet就是使用了模板方法模式。

库设计:在库中,模板方法模式用于定义通用的算法,以满足不同的客户需求。客户可以通过继承库中的类并实现抽象方法来自定义功能。

游戏开发:在游戏开发中,模板方法模式可用于定义游戏中的角色行为或关卡设计,其中不同的角色或关卡可以通过继承来实现特定的行为。

7 优缺点

优点:

代码复用:模板方法模式提供了一种代码复用的方式,使得多个子类可以共享算法的核心结构,减少了重复代码的数量。

扩展性:模板方法模式允许子类扩展或修改算法的特定步骤,而不需要改变算法的整体结构。

高层控制:模板方法模式将算法的控制权交给了抽象类,使得高层模块可以更方便地控制算法的执行。

缺点:

过度抽象:如果不合理地设计抽象类和抽象方法,可能会导致过度抽象,使得代码难以理解和维护。

限制子类:模板方法模式要求子类必须按照模板方法的顺序执行算法步骤,这可能会限制子类的灵活性。

8 类似模式

与模板方法模式类似的模式包括以下几种:

策略模式(Strategy Pattern):策略模式也允许在运行时选择算法,但与模板方法模式不同,它将算法封装成独立的对象,使得客户可以在运行时切换不同的算法实现。模板方法模式通过继承来实现算法的变化,而策略模式通过组合和委托来实现。两者都涉及到将算法进行抽象,但模板方法模式更适用于定义算法的骨架,而策略模式更适用于在不同算法之间进行动态切换。

工厂方法模式(Factory Method Pattern):工厂方法模式用于创建对象,它定义一个创建对象的接口,但让子类决定实例化哪个类。虽然工厂方法模式通常不涉及算法的执行顺序,但它也可以看作是一种创建对象的模板,具有一定的相似性。两者都涉及创建对象,但工厂方法模式关注对象的创建,而模板方法模式关注定义算法的骨架。

命令模式(Command Pattern):命令模式将请求封装成对象,使得可以在不同的上下文中执行请求。虽然命令模式主要关注将请求和执行解耦,但在一些情况下,可以使用模板方法模式来定义具体的命令执行流程。两者都涉及定义执行流程,但命令模式主要用于将请求和执行解耦,而模板方法模式主要用于定义算法的骨架。

虽然这些模式有一些相似之处,但它们的主要关注点和使用方式略有不同。模板方法模式的主要目的是定义算法的骨架,允许子类定制特定的步骤,而其他模式更侧重于其他领域,如创建对象、封装请求等。因此,在选择使用哪种模式时,需要根据具体的需求和问题来决定哪种模式更适合。

9 小结

模板方法模式是一种强大的设计模式,它允许我们定义算法的骨架并延迟特定步骤的实现到子类中。通过这种方式,我们可以实现代码的复用和扩展,同时保持算法的整体结构不变。在实际应用中,模板方法模式常用于框架和库的设计,以及需要定义多个具有相似结构的算法的场景。但要谨慎使用,避免过度抽象和限制子类的问题。在正确的情况下,模板方法模式可以提高代码的可维护性和灵活性,使软件更易于扩展和维护。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/94076.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

Unity调用API函数对系统桌面和窗口截图

Unity3D调用WINAPI函数对系统窗口截图 引入WINAPI函数调用WINAPI函数进行截图使用例子 引入WINAPI函数 using System; using System.Collections; using System.Runtime.InteropServices; using System.Drawing;[DllImport("user32.dll")]private static extern Int…

JAVA 泛型、序列化和复制

泛型提供了编译时类型安全检测机制,该机制允许程序员在编译时检测到非法的类型。泛型的本质是参数化类型,也就是说所操作的数据类型被指定为一个参数。比如我们要写一个排序方法,能够对整型数组、字符串数组甚至其他任何类型的数组进行排序&a…

计算机网络(五):运输层

参考引用 计算机网络微课堂-湖科大教书匠计算机网络(第7版)-谢希仁 1. 运输层概述 之前所介绍的计算机网络体系结构中的物理层、数据链路层以及网络层它们共同解决了将主机通过异构网络互联起来所面临的问题,实现了主机到主机的通信&#xff…

es的nested查询

一、一层嵌套 mapping: PUT /nested_example {"mappings": {"properties": {"name": {"type": "text"},"books": {"type": "nested","properties": {"title": {"t…

响应式Web设计:移动优先的方法

在当今数字时代,移动设备已经成为人们日常生活中不可或缺的一部分。从智能手机到平板电脑,人们几乎无时无刻不在使用移动设备来浏览网站、查看社交媒体和购物。因此,对于博主和网站所有者来说,确保他们的网站在各种屏幕大小和设备…

AtCoder Beginner Contest 233 (A-Ex)

A.根据题意模拟即可 B.根据题意模拟即可 C.直接用map 进行dp即可 D.用前缀和进行模拟,用map统计前缀和,每次计算当前前缀和-k的个数就是以当前点为右端点答案。 E - Σ[k0..10^100]floor(X/10^k) (atcoder.jp) (1)…

Java 多输入框查询需求实现

💗wei_shuo的个人主页 💫wei_shuo的学习社区 🌐Hello World ! 多输入框查询 需求分析 任意一个输入框,输入内容点击搜索都可以精准搜索到对应的内容 代码实现 Controller接口编写 PostMapping("merchant/manage&…

分享Arduino环境下加速下载 第三方库或芯片包

Content 问题描述问题解决 问题描述 众所周知,由于网络的问题,导致Arduino里面的包下载速度非常慢,甚至下了非常久,最后也还是出现下载失败的情况。 有的人打开了加速器,但是也依旧是速度非常慢,为什么呢…

FFmpeg 基础模块:下载编译与安装、常用命令、处理流程

FFmpeg源码下载 我们会逐步分析作为 API 用户我们需要了解的 FFmpeg 中的重要模块,比如 AVFormat 模块、AVcodec 模块、AVfilter 模块、swscale 模块、swresample 模块。 在具体讲解如何使用 FFmpeg 的 API 之前,为了方便你查看 API 对应的代码&#x…

Kafka日志索引详解以及生产常见问题分析与总结

文章目录 1、Kafka的Log日志梳理1.1、Topic下的消息是如何存储的?1.1.1、 log文件追加记录所有消息1.1.2、 index和timeindex加速读取log消息日志。 1.2、文件清理机制1.2.1、如何判断哪些日志文件过期了1.2.2、过期的日志文件如何处理 1.3、Kafka的文件高效读写机制…

使用UE4 HttpRequest提交多表单

大部分HTTP库都是支持直接设置多表单字段的&#xff0c;但UE4的HttpRequest比较惨&#xff0c;只能用SetContent设置整个的TArray<uint8>作为请求体&#xff0c;所以想要传多表单就要自己拼。 首先设置Header&#xff0c;Content-Type设置为多表单&#xff0c;并设置bou…

不做静态化,当部署到服务器上的项目刷新出现404【已解决】

当线上项目刷新出现404页面解决方法&#xff1a; 在nginx配置里加入这样一段代码 try_files $uri $uri/ /index.html; 它的作用是尝试按照给定的顺序访问文件 变量解释 try_files 固定语法 $uri 指代home文件(ip地址后面的路径&#xff0c;假如是127.0.0.1/index/a.png&…

OpenGLES:绘制一个混色旋转的3D圆柱

一.概述 上一篇博文讲解了怎么绘制一个混色旋转的立方体 这一篇讲解怎么绘制一个混色旋转的圆柱 圆柱的顶点创建主要基于2D圆进行扩展&#xff0c;与立方体没有相似之处 圆柱绘制的关键点就是将圆柱拆解成&#xff1a;两个Z坐标不为0的圆 一个长方形的圆柱面 绘制2D圆的…

【Java每日一题】— —第二十题:杨辉三角(直角三角形)。(2023.10.04)

&#x1f578;️Hollow&#xff0c;各位小伙伴&#xff0c;今天我们要做的是第二十题。 &#x1f3af;问题&#xff1a; 杨辉三角&#xff08;直角三角形&#xff09;。 解法1 第一步:动态初始化 第二步:为主对角线及第一列的元素赋值1 第三…

【kylin】【ubuntu】搭建本地源

文章目录 一、制作一个本地源仓库制作ubuntu本地仓库制作kylin本地源 二、制作内网源服务器ubuntu系统kylin系统 三、使用内网源ubuntukylin 一、制作一个本地源仓库 制作ubuntu本地仓库 首先需要构建一个本地仓库&#xff0c;用来存放软件包 mkdir -p /path/to/localname/pac…

嵌入式Linux应用开发-驱动大全-同步与互斥②

嵌入式Linux应用开发-驱动大全-同步与互斥② 第一章 同步与互斥②1.3 原子操作的实现原理与使用1.3.1 原子变量的内核操作函数1.3.2 原子变量的内核实现1.3.2.1 ATOMIC_OP在 UP系统中的实现1.3.2.2 ATOMIC_OP在 SMP系统中的实现 1.3.3 原子变量使用案例1.3.4 原子位介绍1.3.4.1…

tf.compat.v1.global_variables()

tf.global_variables tf.global_variables() 是 TensorFlow 1.x 中的一个函数&#xff0c;它返回图中所有的全局变量。在 TensorFlow 2.x 中&#xff0c;这个函数已经被移除了&#xff0c;取而代之的是 tf.compat.v1.global_variables()。 然而&#xff0c;在 TensorFlow 2.x …

PWN Test_your_nc Write UP

目录 PWN 00 解题过程 总结归纳 PWN 01 解题过程 总结归纳 PWN 02 解题过程 总结归纳 PWN 03 解题过程 总结归纳 PWN 04 解题过程 总结归纳 CTF PWN 开始&#xff01; 冲就完了 PWN 00 解题过程 ssh远程链连接 ssh ctfshowpwn.challenge.ctf.show -p28151 输…

Springboot学习笔记——1

Springboot学习笔记——1 一、快速上手Springboot1.1、Springboot入门程序开发1.1.1、IDEA创建Springboot项目1.1.2、官网创建Springboot项目1.1.3、阿里云创建Springboot项目1.1.4、手工制作Springboot项目 1.2、隐藏文件或文件夹1.3、入门案例解析1.3.1、parent1.3.2、starte…

分布式事务-TCC案例分析流程图

防止cancel方法在最后执行出现问题&#xff0c;用户收到提示已经退款成功但是由于cancel过慢或者出现问题&#xff08;虽然最后会重试成功但是用户体验很差&#xff09;&#xff0c;可以做以下的业务sql模型优化(增加一个冻结金额)。