设计模式【行为型】-- 模板方法模式

news2025/1/23 5:02:01

模板方法模式(Template method pattern)

      模板方法模式是一种行为型设计模式,它定义了一个操作中的算法骨架,将一些步骤的具体实现延迟到子类中。模板方法使得子类可以在不改变算法结构的情况下,重新定义算法中的某些步骤。

概念理论:

模板方法模式基于一个抽象类或接口定义算法的骨架,该骨架由一个模板方法和一些抽象方法组成。模板方法中定义了算法的结构,而抽象方法则由具体子类来实现。子类可以通过重写抽象方法来实现特定步骤的具体行为,而不需要修改算法的整体结构。

优点:

  1. 提供了一种代码复用的方式,将相同的算法结构封装在模板方法中,减少了重复代码的编写。
  2. 提高了代码的可维护性,由于算法的结构被封装在模板方法中,如果需要修改算法的结构,只需要修改模板方法即可,不需要修改每个子类的实现。
  3. 提供了一种扩展的方式,通过在子类中重写抽象方法,可以灵活地改变算法的具体实现。

缺点:

  1. 模板方法模式可能会导致类的个数增加,因为每个具体实现都需要一个子类。
  2. 算法的结构在父类中固定,子类只能通过重写抽象方法来改变算法的具体实现,有一定的局限性。

适用场景:

  1. 当存在一些具有相同算法结构但具体实现不同的操作时,可以使用模板方法模式,将相同的算法结构封装在模板方法中,具体实现由子类提供。
  2. 当需要控制算法的流程,但某些步骤的具体实现可能会变化时,模板方法模式可以提供一种灵活的方式。

案例解析

下面是一个使用模板方法模式的Java示例:

// 抽象类定义算法的骨架
abstract class AbstractClass {
    public void templateMethod() {
        // 步骤1
        step1();

        // 步骤2
        step2();

        // 步骤3
        step3();
    }

    protected abstract void step1();

    protected abstract void step2();

    protected abstract void step3();
}

// 具体子类实现具体的步骤
class ConcreteClass extends AbstractClass {
    @Override
    protected void step1() {
        System.out.println("执行步骤1");
    }

    @Override
    protected void step2() {
        System.out.println("执行步骤2");
    }

    @Override
    protected void step3() {
        System.out.println("执行步骤3");
    }
}

// 使用示例
public class Main {
    public static void main(String[] args) {
        AbstractClass abstractClass = new ConcreteClass();
        abstractClass.templateMethod();
    }
}

在上述代码中,AbstractClass 是抽象类,定义了一个模板方法 templateMethod 和三个抽象方法 step1step2step3templateMethod 方法中按照固定的顺序调用了这三个抽象方法。

ConcreteClass 是具体子类,继承自 AbstractClass,实现了具体的步骤。

Main 类的示例中,我们创建了一个 ConcreteClass 对象,并调用其 templateMethod 方法。可以看到,具体步骤的实现在子类中完成,而算法的结构由父类控制。

模板方法模式适用于存在相同算法结构但具体实现不同的场景,可以提供一种代码复用和扩展的方式。它可以用于控制算法的流程,并在子类中灵活地定义具体的实现。

疑问:templateMethod() 方法使用final 关键字修饰 是不是更合理一些?

在模板方法模式中,将templateMethod()方法使用final关键字修饰是一种常见的做法,可以确保算法的结构不被子类修改。使用final关键字修饰templateMethod()方法可以防止子类对整个算法的修改,而只允许子类通过实现抽象方法来改变算法的具体实现细节。

使用final关键字修饰templateMethod()方法的优点包括:

  1. 确保算法的结构不被修改:通过使用final关键字修饰templateMethod()方法,父类的算法结构变得不可修改,从而保证了算法的一致性和稳定性。
  2. 避免子类的误操作:防止子类意外地修改整个算法的流程,避免了潜在的错误和不一致性。
  3. 提高代码可读性和可维护性:通过使用final关键字明确表示算法的结构不可修改,可以使代码更加清晰和易于理解。

然而,需要注意的是,如果你期望子类能够修改整个算法的结构,或者允许在模板方法中添加新的步骤,那么就不应该使用final关键字修饰templateMethod()方法。在这种情况下,你可以将templateMethod()方法定义为非final,并提供一些钩子方法(hook methods)供子类进行扩展。

综上所述:

  • 使用final关键字修饰templateMethod()方法可以确保算法的结构稳定性和一致性
  • 但需要根据具体需求进行决策,如果希望允许子类修改整个算法结构或添加新的步骤,就不应该使用final关键字修饰。

当我们希望在模板方法模式中允许子类修改整个算法的结构或添加新的步骤时,可以通过提供钩子方法(hook methods)来实现这种扩展性。钩子方法是在模板方法中的空方法或默认实现,子类可以选择性地覆盖或扩展这些方法。

以下是一个结合钩子方法的模板方法模式的Java示例:

// 抽象类定义算法的骨架
abstract class AbstractClass {
    public final void templateMethod() {
        step1();
        step2();

        // 钩子方法
        if (hookMethod()) {
            additionalStep();
        }

        step3();
        step4();
    }

    private void step4(){
        System.out.println("任务执行完毕!");
    }

    protected abstract void step1();

    protected abstract void step2();

    // 钩子方法
    protected boolean hookMethod() {
        return true;
    }

    // 钩子方法的默认实现
    protected void additionalStep() {
        System.out.println("执行附加步骤");
    }

    protected abstract void step3();
}

// 具体子类实现具体的步骤
class ConcreteClass extends AbstractClass {
    @Override
    protected void step1() {
        System.out.println("执行步骤1");
    }

    @Override
    protected void step2() {
        System.out.println("执行步骤2");
    }

    // 重写钩子方法
    @Override
    protected boolean hookMethod() {
        return true; // 在子类中禁用附加步骤
    }

    @Override
    protected void step3() {
        System.out.println("执行步骤3");
    }

    @Override
    protected void additionalStep() {
        System.out.println("我是自定义的扩展");
    }
}

public class TemplateMethodPattern {
    public static void main(String[] args) {
        AbstractClass abstractClass = new ConcreteClass();
        abstractClass.templateMethod();
    }
}

在这里插入图片描述

在上述代码中,我们在抽象类 AbstractClass 中定义了一个钩子方法 hookMethod() 和一个钩子方法的默认实现 additionalStep()。在模板方法 templateMethod() 中,通过调用钩子方法来决定是否执行附加步骤。

在具体子类 ConcreteClass 中,我们重写了钩子方法 hookMethod(),并返回true开启了附加步骤,并且我们还重写了additionalStep() ,这样,当我们调用 templateMethod() 时,就会执行(自定义/默认)附加步骤。(当然我们也可以重写钩子方法 hookMethod(),返回 false 来禁用附加步骤)

通过使用钩子方法,子类可以选择性地扩展算法的结构。在这个示例中,我们通过重写钩子方法来禁用附加步骤,但在其他子类中,可以根据需要重写钩子方法,添加自定义的附加步骤。

这种结构允许子类通过覆盖或扩展钩子方法来自定义算法的行为,从而提供了更大的灵活性和扩展性。

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

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

相关文章

hydra详解(仅供学习参考)

一、概述。 Hydra是一款非常强大的渗透工具,由著名的黑客组织THC开发的一款开源工具。 二、使用方法。 hybra基础语法: hydra 参数 IP 服务 参数: -l login 小写,指定用户名进行破解 -L file 大写,指定用户的用户名…

uniapp uni实人认证

uni实人认证依赖 目前仅支持App平台。 h5端活体人脸检测,使用的是百度云的h5人脸实名认证 使用要求 1、app端 在使用前,请确保您已注册DCloud账号,并已完成实名认证。 然后需要按文档开通服务 业务开通 | uni-app官网 2、h5端 在使用前…

工程系统管理 工程项目管理系统源码 工程项目各模块及其功能点清单

工程项目各模块及其功能点清单 一、系统管理 1、数据字典:实现对数据字典标签的增删改查操作 2、编码管理:实现对系统编码的增删改查操作 3、用户管理:管理和查看用户角色 4、菜单管理:实现对系统菜单的增删改查操…

FFmpeg、x264以及fdk-aac 编译整合

接上文 FFMPEG 编译流程(极客版) 编译 fdk-aac libfdk-aac version:0.1.5 下载 wget http://jaist.dl.sourceforge.net/project/opencore-amr/fdk-aac/fdk-aac-0.1.5.tar.gz#!/bin/bash NDK/home/maqi/Desktop/android-ndk-r20b # 这里需要替换成你本地的 NDK 路径&#x…

【来不及刷题之】43、最小栈(PriorityQueue)

因为要在常量时间内查询出最小值&#xff0c;所以需要有另外的数据结构维护最小值&#xff0c;很自然地想到了“堆”这个结构&#xff0c;“最小堆”的堆顶元素刚好是最小值因此出栈和入栈的同时也要维护好最小堆 class MinStack {PriorityQueue<Integer> heap;LinkedLi…

软件测试前途如何?要学吗?

1.前言 当我们面临择业问题的时候&#xff0c;因为我们本身对自己认知的不清晰和对现有自己能选择的岗位的不了解&#xff0c;往往不知道如何选择才是最优解。这个博客就专门来解答大家对于软件测试这个岗位的疑惑&#xff0c;让大家对软件测试这个岗位有更广义的了解。 本博…

观察者模式(下):如何实现一个异步非阻塞的EventBus框架?

上一节课中&#xff0c;我们学习了观察者模式的原理、实现、应用场景&#xff0c;重点介绍了不同应用场景下&#xff0c;几种不同的实现方式&#xff0c;包括&#xff1a;同步阻塞、异步非阻塞、进程内、进程间的实现方式。 同步阻塞是最经典的实现方式&#xff0c;主要是为了…

C++之std::is_same用法(一百五十八)

简介&#xff1a; CSDN博客专家&#xff0c;专注Android/Linux系统&#xff0c;分享多mic语音方案、音视频、编解码等技术&#xff0c;与大家一起成长&#xff01; 优质专栏&#xff1a;Audio工程师进阶系列【原创干货持续更新中……】&#x1f680; 人生格言&#xff1a; 人生…

Flink HA方案介绍

1.Flink HA方案介绍 每个Flink集群只有单个JobManager&#xff0c;存在单点失败的情况。Flink有YARN、Standalone和Local三种模式&#xff0c;其中YARN和Standalone是集群模式&#xff0c;Local是指单机模式。但Flink对于YARN模式和Standalone模式提供HA机制&#xff0c;使集群…

【历史上的今天】7 月 12 日:世界上第一台商用数字计算机;Python 之父卸任 BDFL;Wacom 成立

整理 | 王启隆 透过「历史上的今天」&#xff0c;从过去看未来&#xff0c;从现在亦可以改变未来。 今天是 2023 年 7 月 12 日&#xff0c;在 1854 年的今天&#xff0c;伊士曼柯达公司的创始人、胶卷的发明者乔治伊斯曼&#xff08;George Eastman &#xff09;出生&#xf…

NC55 最长公共前缀

import java.util.*;public class Solution {/*** 代码中的类名、方法名、参数名已经指定&#xff0c;请勿修改&#xff0c;直接返回方法规定的值即可*** param strs string字符串一维数组* return string字符串*/public String longestCommonPrefix (String[] strs) {int n s…

类和对象(—)

今天&#xff0c;我带来类和对象的基础篇。 目录 面向对象和面向过程类类的概念类的定义类的访问限定符c中struct和class的区别封装类的作用域类的实例化类的存储结构体内存对齐规则【面试题】this指针this指针的概念this指针的特性 【面试题】 面向对象和面向过程 C语言是面向…

[PyTorch][chapter 44][时间序列表示方法3]

简介: word2vec 是 Google 于 2013 年开源推出的一个用于获取 word vector 的工具包&#xff0c;它简单、高效&#xff0c;因此引起了很多人的关注。由于 word2vec 的作者 Tomas Mikolov 其主要知识点 目录&#xff1a; word2vec 基本思想 Skip-gram cbow Hierarchical sof…

MedNeXt的一些问题集锦

归纳偏差是一种关于机器学习算法的目标函数的假设&#xff0c;也就是目标函数评分的标准。 归纳偏差是指模型更容易学习到训练数据中的局部和表面特征&#xff0c;而较难捕捉全局和抽象特征。 scalable 可扩展的 network-wide优势&#xff1f;&#xff1f;&#xff1f; 深度监…

(论文精读)PRUNING FILTER IN FILTER《滤波器中的剪枝滤波器》

论文地址&#xff1a;原文 代码实现 中文翻译 一、精读论文 论文题目 PRUNING FILTER IN FILTER 论文作者 Fanxu Meng 孟繁续 刊物名称 NeurIPS 2020 出版日期 2020 摘要 剪枝已成为现代神经网络压缩和加速的一种非常有效的技术。现有的剪枝方法可分为两大类:滤波器…

MVC三层架构

1.MVC三层架构 MVC&#xff08;Model-View-Controller&#xff09;是一种常见的软件设计模式&#xff0c;用于组织和管理应用程序的代码和逻辑。它将应用程序分为三个主要部分&#xff1a;模型&#xff08;Model&#xff09;、视图&#xff08;View&#xff09;和控制器&#…

营销同质化,博鱼 sports牵手那不勒斯打开新大门

体育营销是企业进入新市场的经典方式&#xff0c;特别是对当今寻求高质量发展的国产品牌而言&#xff0c;从产品出海升级为品牌出海&#xff0c;体育营销可谓是一条必经之路。海信、OPPO、华为等中国品牌通过持续的体育营销不断拉近自身与海外消费者的距离&#xff0c;成功在海…

反常积分定义

目录 反常积分的定义 判断敛散性的方法 方法2&#xff1a; 例题 无界函数的反常积分 判断敛散性的方法 例题 反常积分的定义 该极限存在就表示该反常积分收敛 对于定义3&#xff0c;只有两个都收敛的情况下&#xff0c;原反常积分才收敛。 判断敛散性的方法 始终大的函数形成…

走进USB的U1模式

综述&#xff1a; PCIE有PM和ASPM两种功耗管理模式&#xff0c;USB只有一种 USB有U1/U2/U3三种低功耗模式 本文只针对U1进行分析 如下图所示&#xff0c;为主要状态变换 背景知识 U1是一种低功耗模式&#xff0c;定义的是link的状态不是设备的状态发送LGO_X进入低功耗模式&a…

C++图形开发(12):随机方块的速度和高度

文章目录 1.随机高度2.随机速度3.整段代码4.总结 1.随机高度 那既然是随机&#xff0c;自然少不了随机函数rand()咯~ 详见&#xff1a;C爱好者的自我修养&#xff08;17&#xff09;:rand()随机函数 那么随机速度就可以是&#xff1a; rect_height rand() % int(height / 4)…