23种设计模式-模板方法(Template Method)设计模式

文章目录

  • 一.什么是模板方法模式?
  • 二.模板方法模式的特点
  • 三.模板方法模式的结构
  • 四.模板方法模式的应用场景
  • 五.模板方法模式的优缺点
  • 六.模板方法模式的C++实现
  • 七.模板方法模式的JAVA实现
  • 八.代码解析
  • 九.总结

类图: 模板方法设计模式类图

一.什么是模板方法模式?

模板方法模式(Template Method Pattern)是一种行为型设计模式,它定义了一个操作中的算法骨架,而将一些步骤的实现延迟到子类中。通过这种方式,模板方法允许子类在不改变算法结构的情况下重新定义某些步骤的具体实现。

二.模板方法模式的特点

  1. 算法骨架固定:整体流程在父类中定义,子类只需实现特定步骤。
  2. 代码复用:通用的算法步骤可以在父类中实现,避免重复代码。
  3. 扩展灵活:新增子类时,只需重写相关步骤,无需修改已有代码。

三.模板方法模式的结构

  1. AbstractClass(抽象类)
    • 定义算法的骨架(TemplateMethod),包括一系列步骤(PrimitiveOperation)。
    • 部分步骤由子类实现(抽象方法)。
  2. ConcreteClass(具体类)
    • 实现抽象类定义的抽象方法。
    • 可以根据需要扩展或修改步骤的具体实现。
      模板方法类图

四.模板方法模式的应用场景

  1. 多个子类有共同行为逻辑,但具体实现不同。
  2. 固定的流程需要扩展某些步骤
  3. 避免重复代码,提取通用逻辑。

五.模板方法模式的优缺点

  • 优点
    • 提高代码复用性。
    • 强制遵循一致的算法框架。
    • 易于维护和扩展。
  • 缺点
    • 子类数目较多时,代码可能变得复杂。
    • 设计要求较高,需要仔细分析算法流程。

六.模板方法模式的C++实现

#include <iostream>
using namespace std;

// 抽象类 AbstractClass
class AbstractClass {
public:
    // 模板方法,定义算法骨架
    void TemplateMethod() {
        PrimitiveOperation1(); // 第一步
        PrimitiveOperation2(); // 第二步
        Hook();                // 钩子方法(可选)
    }

    // 基本操作,子类必须实现
    virtual void PrimitiveOperation1() = 0;
    virtual void PrimitiveOperation2() = 0;

    // 钩子方法,子类可以选择性重写
    virtual void Hook() {
        cout << "AbstractClass: Default Hook implementation.\n";
    }

    virtual ~AbstractClass() = default;
};

// 具体类 ConcreteClass
class ConcreteClass : public AbstractClass {
public:
    void PrimitiveOperation1() override {
        cout << "ConcreteClass: Implementing PrimitiveOperation1.\n";
    }

    void PrimitiveOperation2() override {
        cout << "ConcreteClass: Implementing PrimitiveOperation2.\n";
    }

    void Hook() override {
        cout << "ConcreteClass: Overriding Hook method.\n";
    }
};

// 客户端代码
int main() {
    cout << "Using Template Method Pattern:\n";
    AbstractClass* obj = new ConcreteClass();
    obj->TemplateMethod(); // 调用模板方法
    delete obj;
    return 0;
}

七.模板方法模式的JAVA实现

// 抽象类
abstract class AbstractClass {
    // 模板方法
    public final void TemplateMethod() {
        PrimitiveOperation1();
        PrimitiveOperation2();
    }

    // 抽象基本操作(必须由子类实现)
    protected abstract void PrimitiveOperation1();
    protected abstract void PrimitiveOperation2();
}

// 具体类A
class ConcreteClassA extends AbstractClass {
    @Override
    protected void PrimitiveOperation1() {
        System.out.println("ConcreteClassA: Performing PrimitiveOperation1");
    }

    @Override
    protected void PrimitiveOperation2() {
        System.out.println("ConcreteClassA: Performing PrimitiveOperation2");
    }
}

// 具体类B
class ConcreteClassB extends AbstractClass {
    @Override
    protected void PrimitiveOperation1() {
        System.out.println("ConcreteClassB: Performing PrimitiveOperation1");
    }

    @Override
    protected void PrimitiveOperation2() {
        System.out.println("ConcreteClassB: Performing PrimitiveOperation2");
    }
}

// 客户端代码
public class TemplateMethodPatternDemo {
    public static void main(String[] args) {
        // 使用 ConcreteClassA
        AbstractClass classA = new ConcreteClassA();
        System.out.println("ConcreteClassA Template Method Execution:");
        classA.TemplateMethod();

        // 使用 ConcreteClassB
        AbstractClass classB = new ConcreteClassB();
        System.out.println("\nConcreteClassB Template Method Execution:");
        classB.TemplateMethod();
    }
}

八.代码解析

  1. 抽象类
    • TemplateMethod() 是模板方法,定义了算法的执行顺序。
    • PrimitiveOperation1 和 PrimitiveOperation2 是抽象方法,具体实现由子类完成。
    • Hook 是钩子方法,提供默认实现,子类可以选择性重写。
  2. 具体类
    • ConcreteClass 实现了抽象类的抽象方法,并可根据需要覆盖钩子方法。
  3. 客户端代码
    • 创建一个具体类的实例,通过抽象类指针调用模板方法。

九.总结

 模板方法模式通过定义一个算法的通用骨架,为子类提供了灵活的扩展点。它适用于多个子类具有相似逻辑但部分实现不同的场景。通过这种模式,代码的复用性和扩展性得到了极大提升,同时保证了算法逻辑的一致性。

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

uniapp实现开发遇到过的问题(持续更新中....)

1. 在ios模拟器上会出现底部留白的情况 解决方案&#xff1a; 在manifest.json文件&#xff0c;找到开源码视图配置&#xff0c;添加如下&#xff1a; "app-plus" : {"safearea":{"bottom":{"offset" : "none" // 底部安…

Python Matplotlib 安装指南:使用 Miniconda 实现跨 Linux、macOS 和 Windows 平台安装

Python Matplotlib 安装指南&#xff1a;使用 Miniconda 实现跨 Linux、macOS 和 Windows 平台安装 Matplotlib是Python最常用的数据可视化工具之一&#xff0c;结合Miniconda可以轻松管理安装和依赖项。在这篇文章中&#xff0c;我们将详细介绍如何使用Miniconda在Linux、mac…

【element-tiptap】Tiptap编辑器核心概念----结构篇

core-concepts 前言&#xff1a;这篇文章来介绍一下 Tiptap 编辑器的一些核心概念 &#xff08;一&#xff09;结构 1、 Schemas 定义文档组成方式。一个文档就是标题、段落以及其他的节点组成的一棵树。 每一个 ProseMirror 的文档都有一个与之相关联的 schema&#xff0c;…

window的wsl(Ubuntu)安装kafka步骤

环境&#xff1a;Win11 WSL(Linux子系统Ubuntu) apache-zookeeper-3.9.3-bin kafka_2.12-3.8.1 思路&#xff1a;apache上分别下载zookeeper和kafka&#xff0c;在wsl环境安装。在kafka上创建消息的topic&#xff0c;发送消息&#xff0c;接受消息&#xff0c;验证是否安…

Notepad++--在开头快速添加行号

原文网址&#xff1a;Notepad--在开头快速添加行号_IT利刃出鞘的博客-CSDN博客 简介 本文介绍Notepad怎样在开头快速添加行号。 需求 原文件 想要的效果 方法 1.添加点号 Alt鼠标左键&#xff0c;从首行选中首列下拉&#xff0c;选中需要添加序号的所有行的首列&#xff…

机器学习基础06_梯度下降

目录 一、为什么使用梯度下降 二、什么是梯度下降 三、为什么要用梯度下降 四、怎么进行梯度下降 1、微分 1.单变量的微分 2.多变量的微分 2、梯度 3、步骤 (1)学习率α (2)梯度(导数)前的负号 4、实例实现 五、sklearn梯度下降 一、为什么使用梯度下降 前面利用正…

《Vue零基础入门教程》第二课:搭建开发环境

往期内容&#xff1a; 《Vue零基础入门教程》第一课&#xff1a;Vue简介 1 搭建开发环境 Vue环境分为两种 不使用构建工具使用构建丁具 首先&#xff0c;我们会介绍 不使用构建工具 的环境,在组件化章节中介绍 使用构建工具 的方式 1) 初始化 使用如下指令初始化 npm i…

【IDEA】解决总是自动导入全部类(.*)问题

文章目录 问题描述解决方法 我是一名立志把细节说清楚的博主&#xff0c;欢迎【关注】&#x1f389; ~ 原创不易&#xff0c; 如果有帮助 &#xff0c;记得【点赞】【收藏】 哦~ ❥(^_-)~ 如有错误、疑惑&#xff0c;欢迎【评论】指正探讨&#xff0c;我会尽可能第一时间回复…

Acme PHP - Let‘s Encrypt

Lets Encrypt是一个于2015年三季度推出的数字证书认证机构&#xff0c;旨在以自动化流程消除手动创建和安装证书的复杂流程&#xff0c;并推广使万维网服务器的加密连接无所不在&#xff0c;为安全网站提供免费的SSL/TLS证书。 使用PHP来更新证书&#xff1a; Acme PHP | Rob…

【Linux清空显存占用】Linux 系统中清理 GPU 显存

操作指令 # 查看NVIDIA GPU状态和进程 nvidia-smi # 查找所有包含"python"的进程 ps -ef grep python # 强制结束进程号为3023的进程 kill -9 3023截图演示 在 Linux 系统中清理 GPU 显存可以采用以下方法&#xff1a; 1. 终止特定进程&#xff08;常用方法&#x…

【网络】网络抓包与协议分析

网络抓包与协议分析 一. 以太网帧格式分析 这是以太网数据帧的基本格式&#xff0c;包含目的地址(6 Byte)、源地址(6 Byte)、类型(2 Byte)、数据(46~1500 Byte)、FCS(4 Byte)。 Mac 地址类型 分为单播地址、组播地址、广播地址。 单播地址&#xff1a;是指第一个字节的最低位…

IC脚本之perl

Perl 是一种功能丰富的计算机程序语言&#xff0c;运行在超过100种计算机平台上。IC flow 的 流传的古老版本大多是也是使用这种语言&#xff0c;这里会对Perl的常用知识点进行总结。 Note: 所有的语句必须以 “ &#xff1b;”结尾&#xff1b;所有的数据必须先定义才可以使…

MEMS硅麦克风应用电子烟雾化产业稳步爬升,耐高温、 防油、防酸、防腐蚀等性能优势和可实现自动化贴片及极高的一致性等特性使其必将成为主流

全球范围内&#xff0c;电子烟行业正处于快速发展的阶段。随着消费者健康意识的提升和对传统烟草制品替代品需求的增加&#xff0c;电子烟市场获得了显著的增长。然而&#xff0c;伴随而来的监管挑战和消费者期待的变化&#xff0c;也促使行业不断进行技术创新和产品优化。特别…

双因子认证:统一运维平台安全管理策略

01双因子认证概述 双因子认证&#xff08;Two-Factor Authentication&#xff0c;简称2FA&#xff09;是一种身份验证机制&#xff0c;它要求用户提供两种不同类型的证据来证明自己的身份。这通常包括用户所知道的&#xff08;如密码&#xff09;、用户所拥有的&#xff08;如…

快慢指针应用---环型链表的应用

1.题目--判断链表是否成环 已经了解了快慢指针的应用原理&#xff0c;引申&#xff1a;用快慢指针去判断链表是否成环。 题解 简而言之&#xff0c;在fast和slow指针遍历的这种情况下&#xff0c;如果链表是成环的&#xff0c;那么在循环遍历了两次后&#xff0c;fast指针就会…

三、计算机视觉_06YOLO基础知识

1、YOLO概述 1.1 定义 YOLO&#xff08;You Only Look Once&#xff09;是一种流行的对象检测和图像分割模型&#xff0c;由华盛顿大学的 Joseph Redmon 和 Ali Farhadi 于 2015 年推出&#xff0c;因其高速和准确性而迅速受到欢迎 在目标检测领域&#xff0c;传统方法&…

Python Matplotlib 数据可视化全面解析:选择它的七大理由与入门简介

Python Matplotlib数据可视化全面解析&#xff1a;选择它的七大理由与入门简介 本文介绍了Matplotlib这一强大而灵活的数据可视化工具&#xff0c;涵盖其基本概念、独特优势以及为何在众多Python绘图库中脱颖而出。Matplotlib具有广泛的社区支持、高度自定义能力、多样的绘图类…

【Spring Boot】用 MyBatis 实现数据的 CRUD

用 MyBatis 实现数据的 CRUD 1.创建项目 & 引入依赖2.实现数据表的自动初始化3.实现实体对象建模4.实现实体和数据表的映射关系5.实现增加、删除、修改和查询功能6.配置分页功能6.1 增加分页支持6.2 创建分页配置类 7.实现分页控制器8.创建分页视图 本篇博客将通过 MyBatis…

数据结构-二叉树_堆

目录 1.二叉树的概念 ​编辑1.1树的概念与结构 1.2树的相关语 1.3 树的表示 2. ⼆叉树 2.1 概念与结构 2.2 特殊的⼆叉树 2.2.2 完全⼆叉树 2.3 ⼆叉树存储结构 2.3.1 顺序结构 2.3.2 链式结构 3. 实现顺序结构⼆叉树 3.2 堆的实现 3.2.2 向下调整算法 1.二叉树的概…

独家原创 | SCI 1区 高创新预测模型!

往期精彩内容&#xff1a; 时序预测&#xff1a;LSTM、ARIMA、Holt-Winters、SARIMA模型的分析与比较 全是干货 | 数据集、学习资料、建模资源分享&#xff01; EMD变体分解效果最好算法——CEEMDAN&#xff08;五&#xff09;-CSDN博客 拒绝信息泄露&#xff01;VMD滚动分…