【设计模式】模板方法模式 在java中的应用
设计模式:
设计模式是对软件设计中普遍存在(反复出现)的各种问题,所提出的解决方案。这个术语是由Erich Gamma等人在1995年的书《设计模式:可复用面向对象软件的基础》中首次引入的。设计模式可以加快开发过程,提供一种通用的、重复使用的、优雅的解决方案,用于在特定的上下文中处理常见的设计问题。
设计模式可以分为三大类:创建型、结构型和行为型。创建型模式关注如何创建对象,结构型模式关注如何组合对象,而行为型模式则关注对象之间的通信。
模板方法模式:
模板方法模式是一种行为设计模式,它在一个方法中定义了一个算法的骨架,并允许子类为一个或多个步骤提供实现。模板方法使得子类可以在不改变算法结构的情况下,重新定义算法的某些步骤。
在模板方法模式中,我们将不变的行为移至父类,将可变的行为留给子类实现。父类中的模板方法可以定义一系列的算法步骤,并且可以提供一个默认的实现。这样,子类在必要时可以覆盖这些方法,但是改变的是步骤的具体实现,而不是步骤的执行顺序。
模板方法模式是一种非常灵活的模式,它可以用来处理许多常见的编程问题,例如代码重用、代码组织和控制复杂性。
模板方法模式的定义:
模板方法模式是一种行为设计模式,它在一个方法中定义了一个算法的骨架,并允许子类为一个或多个步骤提供实现。模板方法使得子类可以在不改变算法结构的情况下,重新定义算法的某些步骤。
模板方法模式的主要特点:
-
代码复用和封装:模板方法模式通过将公共代码提取到父类中,实现了代码的复用。同时,它封装了具体步骤和数据,保护了算法的完整性和安全性。
-
提供统一的接口:父类提供了一个模板方法,定义了算法的骨架。这个模板方法对外提供了统一的接口,使得客户端不需要关心具体的实现细节。
-
支持变化:模板方法模式允许子类重写父类的某些步骤,这使得算法可以在不改变其结构的情况下,有选择地改变其行为。
-
控制子类的扩展:模板方法模式通过在模板方法中预定义钩子函数,可以控制子类的扩展。钩子是一种特殊的方法,它在父类中声明并给出默认实现,子类可以选择是否覆盖它。
-
延迟实现:模板方法模式中的具体步骤可以在子类中实现,这样可以延迟到子类中进行实现。这是所谓的“好莱坞原则”——别打电话给我们,我们会打电话给你。这种原则可以防止“依赖腐败”。
模板方法模式是一种非常灵活的模式,它可以用来处理许多常见的编程问题,例如代码重用、代码组织和控制复杂性。
模板方法模式的基本结构:
模板方法模式主要包含以下几个部分:
-
抽象类(AbstractClass):这个类定义了一系列的方法,包括模板方法和一些基本方法。模板方法定义了算法的骨架,基本方法是算法中的一个步骤,可以是抽象的,也可以有默认的实现。
-
具体类(ConcreteClass):这个类继承抽象类,并实现抽象类中的抽象方法,这些方法是算法中的具体步骤。
UML图解释:
-----------------
| AbstractClass |
-----------------
| +templateMethod() |
| #primitiveOperation1() |
| #primitiveOperation2() |
-----------------^|
-----------------
| ConcreteClass |
-----------------
| #primitiveOperation1() |
| #primitiveOperation2() |
-----------------
在这个UML图中:
-
AbstractClass是一个抽象类,它定义了一个模板方法templateMethod,以及两个基本方法primitiveOperation1和primitiveOperation2。模板方法templateMethod定义了算法的骨架,基本方法primitiveOperation1和primitiveOperation2是算法中的一个步骤,可以是抽象的,也可以有默认的实现。 -
ConcreteClass是一个具体类,它继承了AbstractClass,并实现了primitiveOperation1和primitiveOperation2。这些方法是算法中的具体步骤。
实例背景介绍:
假设我们正在开发一个工具库,其中有一个DataParser类,这个类的任务是读取数据,解析数据,然后处理数据。我们有两种类型的数据源:一种是从文件中读取的数据,另一种是从数据库中读取的数据。尽管读取数据的方式不同,但解析和处理数据的方式是相同的。
实例设计与实现:
首先,我们定义一个抽象的DataParser类,这个类定义了解析数据的模板方法,模板方法中的一些步骤是抽象的,需要在子类中实现。
public abstract class DataParser {// Template methodpublic final void parseDataAndGenerateReport() {readData();processData();writeReport();}// Abstract methodsprotected abstract void readData();protected abstract void processData();// Common methodpublic void writeReport() {System.out.println("General Report Generation");}
}
然后,我们定义两个DataParser的子类:CSVDataParser和DatabaseDataParser,它们分别实现了读取CSV文件和数据库的方法。
public class CSVDataParser extends DataParser {protected void readData() {System.out.println("Reading data from CSV file");}protected void processData() {System.out.println("Processing data from CSV file");}
}public class DatabaseDataParser extends DataParser {protected void readData() {System.out.println("Reading data from database");}protected void processData() {System.out.println("Processing data from database");}
}
代码解析:
在上述代码中,DataParser类定义了一个模板方法parseDataAndGenerateReport(),这个方法包含了读取数据,处理数据和生成报告这三个步骤。其中,readData()和processData()是抽象方法,需要在子类中实现,而writeReport()是一个具体方法,已经在DataParser类中实现。
CSVDataParser和DatabaseDataParser类分别实现了readData()和processData()方法,它们分别读取CSV文件和数据库中的数据,并处理这些数据。
这样,无论我们的数据来自于CSV文件还是数据库,我们都可以使用相同的方式来解析数据和生成报告,这就是模板方法模式的优点。
模板方法模式在Java API的应用:
在Java API中,有许多地方使用了模板方法模式。例如,java.io.InputStream、java.io.OutputStream、java.io.Reader和java.io.Writer的所有非抽象方法,都依赖于这些类的抽象方法。例如,InputStream中的read()方法就是一个模板方法,它依赖于read(byte b[], int off, int len)这个抽象方法。
Java中模板方法模式的实际应用案例:
一个常见的Java模板方法模式的应用是在Android的AsyncTask类中。AsyncTask是Android提供的一个轻量级的异步类,它可以直接继承使用。AsyncTask中定义了一些方法,如onPreExecute()、doInBackground()、onProgressUpdate()和onPostExecute()。其中doInBackground()是抽象方法,必须在子类中实现,其他都是模板方法,有默认实现,可以根据需要在子类中重写。
以下是一个简单的示例:
public class DownloadTask extends AsyncTask<String, Integer, String> {protected void onPreExecute() {// 在主线程执行,用于进行一些界面上的初始化,比如显示一个进度条对话框等。}protected String doInBackground(String... params) {// 在子线程中执行,进行耗时操作,比如下载文件等。可以调用publishProgress方法来更新任务的进度。return "Downloaded file";}protected void onProgressUpdate(Integer... progress) {// 在主线程执行,用于更新进度信息。}protected void onPostExecute(String result) {// 在主线程执行,用于处理doInBackground方法的结果。}
}
在上述代码中,doInBackground()方法是抽象的,必须在子类中实现,其他的方法都有默认实现,可以在子类中重写。这就是典型的模板方法模式。
模板方法模式的优点:
-
代码复用:模板方法模式通过在抽象类中定义算法的骨架,将具体步骤的实现延迟到子类,可以避免代码重复,提高代码复用性。
-
封装不变部分:模板方法模式封装了不变的算法步骤,保证了算法的稳定性和可靠性。
-
提供扩展点:模板方法模式提供了一个很好的扩展点,即在抽象类中定义的抽象方法,子类可以根据需要重写这些方法,以实现更多的功能。
模板方法模式的缺点:
-
类数量增多:由于模板方法模式需要为每一个具体的算法步骤定义一个子类,因此可能会导致类的数量增多。
-
增加了系统的复杂性:模板方法模式虽然可以提高代码的复用性,但是由于需要定义多个子类,可能会增加系统的复杂性。
-
对子类的设计有限制:模板方法模式把基本方法的执行顺序写在模板方法里,这就限制了子类的行为。如果子类需要改变基本方法的执行顺序,那么就需要修改父类的模板方法,这违反了“开闭原则”。
模板方法模式是一种行为设计模式,它定义了一个操作中的算法的骨架,而将一些步骤延迟到子类中。模板方法使得子类可以在不改变算法结构的情况下,重新定义算法的某些步骤。
相关文章:
【设计模式】模板方法模式 在java中的应用
设计模式: 设计模式是对软件设计中普遍存在(反复出现)的各种问题,所提出的解决方案。这个术语是由Erich Gamma等人在1995年的书《设计模式:可复用面向对象软件的基础》中首次引入的。设计模式可以加快开发过程&#x…...
PVE纵览-安装系统卡“Loading Driver”的快速解决方案
PVE纵览-安装系统卡“Loading Driver”的快速解决方案 文章目录 PVE纵览-安装系统卡“Loading Driver”的快速解决方案摘要通过引导参数解决PVE安装卡在“Loading Driver”问题官方解决方法 关键字: PVE、 显卡、 Loading、 Driver、 nomodeset 摘要 在虚拟机…...
Lua资料
Lua脚本语言 cheet sheet Lua & c Lua与C API交互全面解析 Lua语言:和C语言的交互 Lua进阶用法之Lua和C的接口设计 Lua C API 简介 C和Lua之间的相互调用 深入Lua:用户数据userdata 基本数据类型 之 UserData calling-lua-from-c/ Embedding Lua i…...
【C语言】值传递和地址传递
值传递 引用传递(传地址,传引用)的区别 传值,是把实参的值赋值给行参 ,那么对行参的修改,不会影响实参的值。 传地址,是传值的一种特殊方式,只是他传递的是地址,不是普通…...
PyTorch 中使用自动求导计算梯度
使用 PyTorch 进行自动求导和梯度计算 在 PyTorch 中,张量的 requires_grad 属性决定了是否需要计算该张量的梯度。设置为 True 的张量会在计算过程中记录操作,以便在调用 .backward() 方法时自动计算梯度。通过构建计算图,PyTorch 能够有效…...
Oracle Instant Client 23.5安装配置完整教程
Oracle Instant Client 23.5安装配置完整教程 简介环境要求安装步骤1. 准备工作目录2. 下载Oracle Instant Client3. 解压Instant Client4. 安装依赖包5. 配置系统环境5.1 配置库文件路径5.2 配置环境变量 6. 配置Oracle钱包(可选) 验证安装常见问题解决…...
【jvm】方法区的理解
目录 1. 说明2. 方法区的演进3. 内部结构4. 作用5.内存管理 1. 说明 1.方法区用于存储已被虚拟机加载的类信息、常量、静态变量、即时编译器编译后的代码缓存等数据。它是各个线程共享的内存区域。2.尽管《Java虚拟机规范》中把方法区描述为堆的一个逻辑部分,但它却…...
ES-针对某个字段去重后-获取某个字段值的所有值
针对上面表的数据,现在想根据age分组,并获取每个分组后的name有哪些(去重后)。 select age, GROUP_CONCAT(DISTINCT(name)) from testtable group by age ; 结果: 如果想要增加排序: SELECT age, GROUP_CONCAT(DISTINCT name)…...
百度 2025届秋招提前批 文心一言大模型算法工程师
文章目录 个人情况一面/技术面 1h二面/技术面 1h三面/技术面 40min 个人情况 先说一下个人情况: 学校情况:211本中9硕,本硕学校都一般,本硕都是计算机科班,但研究方向并不是NLP,而是图表示学习论文情况&a…...
sglang 部署Qwen2VL7B,大模型部署,速度测试,深度学习
sglang 项目github仓库: https://github.com/sgl-project/sglang 项目说明书: https://sgl-project.github.io/start/install.html 资讯: https://github.com/sgl-project/sgl-learning-materials?tabreadme-ov-file#the-first-sglang…...
fastadmin操作数据库字段为json、查询遍历each、多级下拉、union、php密码设置、common常用函数的使用小技巧
数据库中遇到的操作 查询字段是json的某个值 //获取数据库中某个字段是json中得某个值,进行查询,goods是表中字段,brand_id是json中要查詢的字段。//数据类型一定要对应要不然查询不出来。$map[json_extract(goods, "$.brand_id")]…...
UniApp在Vue3的setup语法糖下自定义组件插槽详解
UniApp在 Vue3的 setup 语法糖下自定义组件插槽详解 UniApp 是一个基于 Vue.js 的跨平台开发框架,可以用来开发微信小程序、H5、App 等多种平台的应用。Vue 3 引入了 <script setup> 语法糖,使得组件的编写更加简洁和直观。本文将详细介绍如何在 …...
springboot上传下载文件
RequestMapping(“bigJson”) RestController Slf4j public class TestBigJsonController { Resource private BigjsonService bigjsonService;PostMapping("uploadJsonFile") public ResponseResult<Long> uploadJsonFile(RequestParam("file")Mul…...
Python学习从0到1 day29 Python 高阶技巧 ⑦ 正则表达式
目录 一、正则表达式 二、正则表达式的三个基础方法 1.match 从头匹配 2.search(匹配规则,被匹配字符串) 3.findall(匹配规则,被匹配字符串) 三、元字符匹配 单字符匹配: 注: 示例&a…...
机器学习-web scraping
Web Scraping,通常称为网络抓取或数据抓取,是一种通过自动化程序从网页中提取数据的技术。以下是对Web Scraping的详细解释: 一、定义与原理 Web Scraping是指采用技术手段从大量网页中提取结构化和非结构化信息,并按照一定的规…...
移远通信5G RedCap模组RG255C-CN通过中国电信5G Inside终端生态认证
近日,移远通信5G RedCap模组RG255C-CN荣获中国电信颁发的5G Inside终端生态认证证书。这表明,该产品在5G基本性能、网络兼容性、安全特性等方面已经过严格评测且表现优异,将进一步加速推动5G行业终端规模化应用。 中国电信5G Inside终端生态认…...
Javaweb梳理17——HTMLCSS简介
Javaweb梳理17——HTML&CSS简介 17 HTML&CSS简介17.1 HTML介绍17.2 快速入门17.3 基础标签17.3 .1 标题标签17.3.2 hr标签17.3.3 字体标签17.3.4 换行17.3.8 案例17.3.9 图片、音频、视频标签17.3.10 超链接标签17.3.11 列表标签17.3.12 表格标签17.3.11 布局标签17.3.…...
【Android、IOS、Flutter、鸿蒙、ReactNative 】自定义View
Android Java 自定义View 步骤 创建一个新的Java类,继承自View、ViewGroup或其他任何一个视图类。 如果需要,重写构造函数以支持不同的初始化方式。 重写onMeasure方法以提供正确的测量逻辑。 重写onDraw方法以实现绘制逻辑。 根据需要重写其他方法&…...
win11跳过联网激活步骤
win11跳过联网激活步骤 win11跳过联网激活步骤方法一:使用Shift F10快捷键(推荐)1. 启动Windows 112. 选择键盘布局或输入法3. 是否想要添加第二种键盘布局4. 让我们为你连接到网络5. 调出管理员模式CMD6. 耐心等待自动重启7. 启动Windows 1…...
利用c语言详细介绍下冒泡排序
软件开发过程中,排序算法是常规且使用众多的方法之一,而冒泡算法又是排序算法中最常规且基本的算法。今天我们利用c语言,图文详细介绍下冒泡算法。 一、图文介绍 我们输入一个数组,数组为【10,5,3…...
ScriptCat脚本猫完整指南:为什么它是浏览器脚本管理的终极选择
ScriptCat脚本猫完整指南:为什么它是浏览器脚本管理的终极选择 【免费下载链接】scriptcat ScriptCat, a browser extension that can execute userscript; 脚本猫,一个可以执行用户脚本的浏览器扩展 项目地址: https://gitcode.com/gh_mirrors/sc/scr…...
数据库的undo和redo日志
本文介绍undo和redo日志的一般概念,不涉及具体某个数据库的实现细节,参考资料来自《数据库系统实现》的第六章《系统故障对策》。一个假设和四个操作原语一个假设假设数据库由元素组成。为了简化讨论,这里假设元素是磁盘块,并且元…...
如何让经典魔兽争霸在现代系统重生?WarcraftHelper兼容性解决方案全解析
如何让经典魔兽争霸在现代系统重生?WarcraftHelper兼容性解决方案全解析 【免费下载链接】WarcraftHelper Warcraft III Helper , support 1.20e, 1.24e, 1.26a, 1.27a, 1.27b 项目地址: https://gitcode.com/gh_mirrors/wa/WarcraftHelper 当你在Windows 11…...
图像矢量化新纪元:Vectorizer从技术原理到实战应用
图像矢量化新纪元:Vectorizer从技术原理到实战应用 【免费下载链接】vectorizer Potrace based multi-colored raster to vector tracer. Inputs PNG/JPG returns SVG 项目地址: https://gitcode.com/gh_mirrors/ve/vectorizer 一、重新定义图像的数字形态&a…...
基于Python的交通数据分析应用2025_mjev917n
前言随着家庭用电需求的日益增长,精准预测用电量对于能源管理与成本控制具有重要意义。通过预测家庭用电量,用户可以更好地规划用电行为,降低能源消耗,实现节能减排的目标。同时,对于电力公司而言,准确的用…...
【JavaScript高级编程】拆解函数流水线 上犯
一、什么是setuptools? setuptools 是一个用于创建、分发和安装 Python 包的核心库。 它可以帮助你: 定义 Python 包的元数据(如名称、版本、作者等)。 声明包的依赖项,确保你的包能够正确运行。 构建源代码分发包&…...
《东方星动》“一路有你”公益行走进云南宾川县上沧完小
春风送暖,爱意流淌。4月3日,云南省大理白族自治州宾川县上沧完小校园内,一场以“爱心筑梦情暖校园”为主题的捐赠仪式温暖举行。这是《东方星动》“一路有你”公益行团队继湖南平江之后,再次跨越千里,将艺术的种子播撒…...
Qwen3Guard-Gen-8B开箱即用:离线内容审核,保护你的AI应用免受风险
Qwen3Guard-Gen-8B开箱即用:离线内容审核,保护你的AI应用免受风险 1. 为什么需要离线内容审核? 在AI应用快速发展的今天,内容安全问题日益突出。无论是社交媒体、在线客服还是内容创作平台,都可能面临以下风险&#…...
OpenClaw多任务调度:千问3.5-9B并行处理技巧
OpenClaw多任务调度:千问3.5-9B并行处理技巧 1. 为什么需要多任务调度 去年冬天,我接手了一个数据密集型项目,需要同时处理数据分析、邮件生成和文件格式转换三项任务。最初尝试用传统脚本串行执行,结果发现总耗时超过8小时——…...
OpenClaw代码审查:Qwen3-14B分析Git提交并标注潜在风险
OpenClaw代码审查:Qwen3-14B分析Git提交并标注潜在风险 1. 为什么需要AI辅助代码审查 作为一个长期维护个人项目的开发者,我经常面临一个尴尬局面:在深夜赶代码时,容易忽略一些基础安全风险。直到某次线上事故后,我开…...
