【设计模式】【行为型模式】模板方法模式(Template Method)

👋hi,我不是一名外包公司的员工,也不会偷吃茶水间的零食,我的梦想是能写高端CRUD
🔥 2025本人正在沉淀中… 博客更新速度++
📫 欢迎+V: flzjcsg2,我们共同讨论Java深渊的奥秘
🎵 当你的天空突然下了大雨,那是我在为你炸乌云

文章目录

  • 一、入门
    • 1.1、什么是模板方法模式?
    • 1.2、为什么要模板方法模式?
      • 1.2.1、代码复用
      • 1.2.2、符合开闭原则
      • 1.2.3、支持框架设计
    • 1.3、怎么实现模板方法模式?
  • 二、模板方法模式在源码中运用
    • 2.1、Java集合
    • 2.2、AQS
  • 三、总结
  • 参考

一、入门

1.1、什么是模板方法模式?

模板模式(Template Method Pattern)是一种行为设计模式,它定义了一个算法的框架,并允许子类在不改变算法结构的情况下重新定义算法的某些步骤。模板模式通过将算法的通用部分放在父类中,而将可变部分留给子类来实现。

1.2、为什么要模板方法模式?

1.2.1、代码复用

模板模式的核心是将算法的通用部分放在父类中实现,而将可变的部分留给子类去实现。通过这种方式,模板模式实现了代码复用,同时提供了灵活性。

1.2.2、符合开闭原则

符合开闭原则,开闭原则(Open/Closed Principle)要求软件实体(类、模块、函数等)对扩展开放,对修改关闭。模板模式通过将可变的部分抽象出来,允许子类扩展算法的某些步骤,而无需修改父类的代码。 这符合开闭原则,提高了系统的可维护性和可扩展性。
示例:
在 Java 的 HttpServlet 中,service() 方法定义了处理 HTTP 请求的框架,但具体的 doGet()doPost() 方法由子类实现。如果需要支持新的 HTTP 方法,只需添加新的方法,而无需修改 service() 方法。

1.2.3、支持框架设计

模板模式在框架设计中非常有用。框架通常定义了一个通用的流程,但具体的实现由开发者完成。模板模式允许框架提供默认实现,同时允许开发者自定义某些步骤。
模板模式为框架设计提供了一种灵活且可扩展的方式。
示例:
在 Spring MVC 中,AbstractController 定义了处理 HTTP 请求的框架,但具体的请求处理逻辑由开发者实现。

1.3、怎么实现模板方法模式?

定义一个抽象类: 写一个抽象类,里面包含一个模板方法(通常是 final 的),用来定义算法的步骤。在模板方法中,调用一些具体方法(有默认实现)和抽象方法(需要子类实现)。
实现具体类:创建一个子类,继承抽象类。实现抽象类中的抽象方法,提供具体的逻辑。
使用模板模式: 创建具体类的对象,调用模板方法。模板方法会自动按照定义好的步骤执行,同时调用子类实现的逻辑。

【案例】炒菜
炒菜的步骤是固定的,分为倒油、热油、倒菜、倒调料、翻炒等步骤。
在这里插入图片描述
AbstractClass类。这个abstract关键字只能在抽象类中,且子类必须实现abstract修饰的方法,就像2楼是盖在1楼上的。final关键字不是必须加的,看实际业务情况,加上后可以避免子类重写该方法。

public abstract class AbstractClass {public final void cookProcess() {//第一步:倒油this.pourOil();//第二步:热油this.heatOil();//第三步:倒蔬菜this.pourVegetable();//第四步:倒调味料this.pourSauce();//第五步:翻炒this.fry();}public void pourOil() {System.out.println("倒油");}//第二步:热油是一样的,所以直接实现public void heatOil() {System.out.println("热油");}//第三步:倒蔬菜是不一样的(一个下包菜,一个是下菜心)public abstract void pourVegetable();//第四步:倒调味料是不一样public abstract void pourSauce();//第五步:翻炒是一样的,所以直接实现public void fry(){System.out.println("炒啊炒啊炒到熟啊");}
}

ConcreteClass_BaoCai 类

public class ConcreteClass_BaoCai extends AbstractClass {@Overridepublic void pourVegetable() {System.out.println("下锅的蔬菜是包菜");}@Overridepublic void pourSauce() {System.out.println("下锅的酱料是辣椒");}
}

ConcreteClass_CaiXin类

public class ConcreteClass_CaiXin extends AbstractClass {@Overridepublic void pourVegetable() {System.out.println("下锅的蔬菜是菜心");}@Overridepublic void pourSauce() {System.out.println("下锅的酱料是蒜蓉");}
}
Clinet
public class Client {public static void main(String[] args) {//炒手撕包菜ConcreteClass_BaoCai baoCai = new ConcreteClass_BaoCai();baoCai.cookProcess();//炒蒜蓉菜心ConcreteClass_CaiXin caiXin = new ConcreteClass_CaiXin();caiXin.cookProcess();}
}

二、模板方法模式在源码中运用

2.1、Java集合

AbstractList 是抽象类,提供了列表和集合的基本能力。其中的通用addAll方法,但是具体的实现add方法的实现,交给了子类

public boolean addAll(int index, Collection<? extends E> c) {rangeCheckForAdd(index);boolean modified = false;for (E e : c) {add(index++, e);     // 子类来实现modified = true;}return modified;
}public void add(int index, E element) {throw new UnsupportedOperationException();
}

ArrayList的实现

public void add(int index, E element) {rangeCheckForAdd(index);ensureCapacityInternal(size + 1);  // Increments modCount!!System.arraycopy(elementData, index, elementData, index + 1,size - index);elementData[index] = element;size++;
}

LinkedList的实现

public void add(int index, E element) {checkPositionIndex(index);if (index == size)linkLast(element);elselinkBefore(element, node(index));
}

2.2、AQS

AQS 主要用于实现锁和同步器,它提供了抽象的框架方法,而具体的同步操作实现(如锁的获取、释放)则由子类来实现。这就是模板方法模式的关键特征:定义一个算法的骨架,将一些步骤延迟到子类中。
这里用tryAcquiretryRelease举例

protected boolean tryAcquire(int arg) {throw new UnsupportedOperationException();
}protected boolean tryRelease(int arg) {throw new UnsupportedOperationException();
}

ReentrantLock 中 Sync类实现。ReentrantLock 是 AQS 的一个典型子类实现,它通过内部类 Sync 实现了tryAcquiretryRelease方法。

protected final boolean tryAcquire(int acquires) {final Thread current = Thread.currentThread();int c = getState(); // 获取当前同步状态if (c == 0) { // 如果状态为0,表示锁未被占用if (!hasQueuedPredecessors() && // 检查是否有线程在等待compareAndSetState(0, acquires)) { // CAS 设置状态setExclusiveOwnerThread(current); // 设置当前线程为独占线程return true;}}else if (current == getExclusiveOwnerThread()) { // 如果是重入int nextc = c + acquires;if (nextc < 0) // 溢出检查throw new Error("Maximum lock count exceeded");setState(nextc); // 更新状态return true;}return false; // 获取锁失败
}protected final boolean tryRelease(int releases) {int c = getState() - releases; // 计算新的状态if (Thread.currentThread() != getExclusiveOwnerThread()) // 检查当前线程是否持有锁throw new IllegalMonitorStateException();boolean free = false;if (c == 0) { // 如果状态为0,表示锁完全释放free = true;setExclusiveOwnerThread(null); // 清除独占线程}setState(c); // 更新状态return free;
}

AQS 在acquirerelease方法中会调用子类实现的tryAcquiretryRelease

public final void acquire(int arg) {if (!tryAcquire(arg) && // 尝试获取锁acquireQueued(addWaiter(Node.EXCLUSIVE), arg)) // 如果失败,加入队列并阻塞selfInterrupt();
}public final boolean release(int arg) {if (tryRelease(arg)) { // 尝试释放锁Node h = head;if (h != null && h.waitStatus != 0)unparkSuccessor(h); // 唤醒队列中的下一个线程return true;}return false;
}

三、总结

模板方法的结构:
● 抽象类(Abstract Class):定义一个算法的框架,包含一个模板方法和一些基本操作(可以是抽象的或已经实现的)。
● 模板方法(Template Method):在抽象类中定义的具体方法,它通过调用其他步骤(基本操作)来实现算法的步骤,子类可以通过覆盖某些步骤来调整具体的行为。
● 具体子类(Concrete Class):继承自抽象类,并实现或重写抽象类中的一些方法。

优点:

  1. 代码复用:模板方法通过将公共的算法框架定义在父类中,减少了子类中重复代码的编写。
  2. 扩展性:子类可以根据需要修改或扩展特定的步骤,而无需改变算法的整体结构。
  3. 控制算法流程:通过模板方法,父类可以控制整个算法的执行流程,确保子类在特定的顺序中执行步骤。
  4. 提高一致性:算法的骨架是固定的,这使得不同的子类可以按照相同的步骤进行操作,增强了程序的可维护性。

缺点:

  1. 增加类的数量:每次修改和扩展都可能需要增加新的子类,这可能会使系统类的数量增加,导致复杂度提高。
  2. 不灵活的继承结构:模板方法的设计依赖于继承,如果需要灵活变化或组合不同的步骤,可能会导致类的设计过于僵化。
  3. 难以适应复杂的变化:当算法变得非常复杂或频繁变化时,可能需要频繁修改父类的模板方法,这可能不太符合开闭原则。

适用场景:

  1. 有多个步骤的算法:当一个操作包含多个步骤,而这些步骤可以通过不同的子类进行调整时,模板方法模式非常适用。例如,游戏中不同角色的攻击动作,或者不同文档生成过程的步骤等。
  2. 有部分固定操作和部分可变操作的场景:当某些步骤是固定的,而其他步骤可能会变化时,模板方法模式可以帮助实现固定部分的复用,并允许子类根据需要重写可变部分。
  3. 不希望改变算法结构时:如果你需要确保子类按照某种固定的顺序执行步骤,但又希望在某些步骤上有所不同,模板方法可以很好地适用。

参考

重学 Java 设计模式:实战模板模式「模拟爬虫各类电商商品,生成营销推广海报场景」

黑马程序员Java设计模式详解, 23种Java设计模式(图解+框架源码分析+实战)_哔哩哔哩_bilibili

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

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

相关文章

基础设施在平台工程中的作用

平台工程侧重于设计和构建自助服务工具和环境&#xff0c;以简化软件开发和部署。通过简化和隐藏底层系统的复杂性&#xff0c;我们可以将精力集中在提供有意义的价值上。 从传统的 IT 运营过渡到集成的 DevOps 基础设施实践优先考虑团队合作、简化的流程和持续交付&#xff0…

Unity3D实现显示模型线框(shader)

系列文章目录 unity工具 文章目录 系列文章目录👉前言👉一、效果展示👉二、第一种方式👉二、第二种方式👉壁纸分享👉总结👉前言 在 Unity 中显示物体线框主要基于图形渲染管线和特定的渲染模式。 要显示物体的线框,通常有两种常见的方法:一种是利用内置的渲染…

活动预告 |【Part1】Microsoft Azure 在线技术公开课:AI 基础知识

课程介绍 参加“Azure 在线技术公开课&#xff1a;AI 基础知识”活动&#xff0c;了解 AI 核心概念。参加我们举办的本次免费培训活动&#xff0c;了解组织如何使用 AI 技术克服实际挑战&#xff0c;以及如何借助 Azure AI 服务构建智能应用程序。本次培训适用于任何对 AI 解决…

Hello Robot 推出Stretch 3移动操作机器人,赋能研究与商业应用

Hello Robot公司近日发布了其新一代开源移动操作机器人Stretch 3&#xff0c;这是一款高度灵活的机器人平台&#xff0c;专为机器人研究、教育实验和商业自动化设计。Stretch 3 结合了先进的移动机器人技术、灵巧操作能力和开源软件生态系统&#xff0c;为用户提供了一个功能强…

题解 洛谷 Luogu P1828 [USACO3.2] 香甜的黄油 Sweet Butter 最短路 堆优化Dijkstra Floyd C++

题目 传送门 P1828 [USACO3.2] 香甜的黄油 Sweet Butter - 洛谷 | 计算机科学教育新生态https://www.luogu.com.cn/problem/P1828 思路 以每头奶牛所在的牧场为起点&#xff0c;求得到全图各个点的最短距离 再枚举全图所有点&#xff0c;计算从所有起点到某点的距离之和&a…

堆排序

目录 堆排序&#xff08;不稳定&#xff09;&#xff1a; 代码实现&#xff1a; 思路分析&#xff1a; 总结&#xff1a; 堆排序&#xff08;不稳定&#xff09;&#xff1a; 如果想要一段数据从小到大进行排序&#xff0c;则要先建立大根堆&#xff0c;因为这样每次堆顶上都能…

2.11日学习总结

题目一 &#xff1a; AC代码 #include <stdio.h> #include <stdlib.h>// 定义长整型 typedef long long ll;// 定义求最大值和最小值的宏函数 #define MAX(a, b) ((a) > (b) ? (a) : (b)) #define MIN(a, b) ((a) < (b) ? (a) : (b))// 定义数组和变量 ll…

Ollama 简单 好用 好玩

简介 Ollama https://github.com/ollama/ollama/ 是一个基于 Go 语言 的 本地大语言模型运行框架&#xff0c;专注于本地化运行大型语言模型&#xff08;LLM&#xff09;的开源工具。 类 Docker 产品&#xff08;支持 list,pull,push,run 等命令&#xff09;&#xff0c;更好玩…

【AIGC】在VSCode中集成 DeepSeek(OPEN AI同理)

在VSCode中集成 DeepSeek&#xff08;OPEN AI同理&#xff09; 一、集成 DeepSeek二、其他推荐VSCode插件 在 Visual Studio Code (VSCode) 中集成 AI 编程能力&#xff0c;可以通过安装和配置特定插件来实现。以下是如何通过 Continue 和 Cline 插件集成 DeepSeek&#xff1a;…

SpringBootWeb三层架构分层解耦

SpringBootWeb 1. SpringBootWeb案例1.1 控制层未拆分代码1.2 实体类1.3 静态资源文件1.4 txt文件1.5 运行界面展示 2. 三层架构拆分2.1 控制层&#xff08;Controller&#xff09;2.1.1 功能2.1.2 用户信息控制层 2.2 业务逻辑层&#xff08;Service&#xff09;2.2.2 功能2.2…

MyBatis的工作流程是怎样的?

大家好&#xff0c;我是锋哥。今天分享关于【MyBatis的工作流程是怎样的&#xff1f;】面试题。希望对大家有帮助&#xff1b; MyBatis的工作流程是怎样的&#xff1f; 1000道 互联网大厂Java工程师 精选面试题-Java资源分享网 MyBatis 的工作流程可以分为几个主要的步骤&…

carbon 加入 GitCode:Golang 时间处理的 “瑞士军刀”

在 Golang 的开发生态中&#xff0c;时间处理领域长期存在着诸多挑战。高效、精准的时间处理对于各类软件应用的稳定运行与功能拓展至关重要。近日&#xff0c;carbon 正式加入 GitCode&#xff0c;为 Golang 开发者带来一款强大且便捷的时间处理利器&#xff0c;助力项目开发迈…

【力扣】148.排序链表

AC截图 题目 思路 基本情况处理&#xff1a; 如果链表为空 (head NULL) 或者链表仅有一个节点 (head->next NULL)&#xff0c;则链表已经是有序的&#xff0c;直接返回头节点 head。 分割链表&#xff1a; 使用快慢指针法找到链表的中间节点。slow 指针每次前进一格&…

新数据结构(7)——Object

Object类是所有类的父类&#xff0c;也就是说所有类都是object类的子类可以使用Object里的方法。 equals()和hashCode()是Object包含的方法 和equals的区别和联系 和equals()都是用于判断是否相等 基本数据类型只能用判断其是否相等&#xff0c;引用数据类型用判断的是其指…

保研考研机试攻略:python笔记(4)

🐨🐨🐨15各类查找 🐼🐼二分法 在我们写程序之前,我们要定义好边界,主要是考虑区间边界的闭开问题。 🐶1、左闭右闭 # 左闭右闭 def search(li, target): h = len(li) - 1l = 0#因为都是闭区间,h和l都可以取到并且相等while h >= l:mid = l + (h - l) // 2…

doris集群

开发doris的团队厉害&#xff0c;这个百度工程师确实也干了一些实事&#xff0c;不像领导层只会跑火车。 1 参数配置 1.1 文件句柄数 vim /etc/security/limits.conf * soft nofile 655350 * hard nofile 6553501.2 关闭透明大页 echo never > /sys/kernel/mm/transpare…

STM32的HAL库开发---高级定时器---互补输出带死区实验

一、互补输出简介 互补输出&#xff1a;OCx输出高电平&#xff0c;则互补通道OCxN输出低电平。OCx输出低电平&#xff0c;则互补通道OCxN输出高电平。 带死区控制的互补输出&#xff1a;OCx输出高电平时&#xff0c;则互补通道OCxN过一会再输出输出低电平。这个时间里输出的电…

京东广告生成式召回基于 NVIDIA TensorRT-LLM 的推理加速实践

0000 生成式推荐系统优势介绍 推荐系统的主要任务是根据用户的历史行为预测其兴趣点&#xff0c;并向其推荐相应的商品。传统的推荐系统在处理用户请求时&#xff0c;会触发多个召回模块&#xff08;包括热门商品召回、个性化召回、深度召回等&#xff09;&#xff0c;以召回大…

3.React 组件化开发

react&#xff1a;版本 18.2.0node&#xff1a; 版本18.19.1脚手架&#xff1a;版本 5.0.1 一、类组件 (一) 一个干净的脚手架 【1】使用已经被废弃的 CRA (create-react-app) create-react-app 已经被废弃&#xff0c;且目前使用会报错&#xff0c;官方已经不推荐使用&…

51单片机(国信长天)矩阵键盘的基本操作

在CT107D单片机综合训练平台上&#xff0c;首先将J5处的跳帽接到1~2引脚&#xff0c;使按键S4~S19按键组成4X4的矩阵键盘。在扫描按键的过程中&#xff0c;发现有按键触发信号后(不做去抖动)&#xff0c;待按键松开后&#xff0c;在数码管的第一位显示相应的数字:从左至右&…