设计模式4-模版方法

news2025/1/16 5:37:42

设计模式

  • 重构获得模式
  • 重构的关键技法
      • 1. 静态转动态
      • 2. 早绑定转晚绑定
      • 3. 继承转组合
      • 4. 编译时依赖转运行时依赖
      • 5. 紧耦合转松耦合
  • 组件协助
    • 动机
    • 模式定义
      • 结构
    • 要点总结。
  • 例子
      • 示例解释:

重构获得模式

设计模式的目的是应对变化,提高复用

设计模式的要点是寻找变化点,然后在变化点处应用设计模式。从而更好地应对需求的变化。什么时候什么地点应用设计模式。比设计模式结构本身更为重要。
设计模式的应用,不应当先入为主,一上来就套用设计模式这是设计模式的最大误用,没有一步到位的设计。敏捷软件开发实践提倡的是refacroring to Patterns(重构), 是目前普遍公认的最好使用的设计模式的方法。

推荐书籍

在这里插入图片描述

重构的关键技法

让变化可配置
重构是改善现有代码设计的过程,通过一系列的技术手法和策略,使得代码更易于理解、扩展和维护。以下是几种重构的关键技法,以及它们在代码中的应用示例:

1. 静态转动态

技法说明

  • 将在编译时确定的部分改为在运行时确定,增加代码的灵活性和适应性。

示例

  • 将硬编码的条件判断改为通过配置文件或者用户输入来确定。
// 静态
bool isFeatureEnabled() {
    return true;  // 静态确定特性总是启用
}

// 动态
bool isFeatureEnabled() {
    // 通过配置文件或者用户输入来确定特性是否启用
    return config.isEnabled("featureX");
}

2. 早绑定转晚绑定

技法说明

  • 将函数调用的绑定从编译时确定延迟到运行时,使得对象能够在运行时选择调用不同的方法或策略。

示例

  • 使用虚函数和多态来实现晚绑定,而不是通过静态绑定的方式。
// 早绑定
class Shape {
public:
    void draw() {
        // 绘制代码
    }
};

// 晚绑定
class Shape {
public:
    virtual void draw() {
        // 绘制代码
    }
};

class Circle : public Shape {
public:
    void draw() override {
        // 绘制圆形的代码
    }
};

class Rectangle : public Shape {
public:
    void draw() override {
        // 绘制矩形的代码
    }
};

void render(Shape* shape) {
    shape->draw();  // 晚绑定,根据实际对象类型调用对应的draw方法
}

3. 继承转组合

技法说明

  • 通过组合现有类的实例来代替继承,以减少类之间的耦合性,同时提高代码的灵活性和可维护性。

示例

  • 将继承关系改为对象组合关系。
// 继承
class Car {
public:
    virtual void drive() {
        // 驾驶汽车
    }
};

class SportsCar : public Car {
public:
    void drive() override {
        // 驾驶运动汽车
    }
};

// 组合
class Engine {
public:
    void start() {
        // 启动引擎
    }
};

class Car {
private:
    Engine engine;

public:
    void drive() {
        engine.start();
        // 驾驶汽车
    }
};

class SportsCar {
private:
    Engine engine;

public:
    void drive() {
        engine.start();
        // 驾驶运动汽车
    }
};

4. 编译时依赖转运行时依赖

技法说明

  • 将程序依赖关系从编译时硬编码转为运行时根据条件动态确定,提高程序的灵活性和可配置性

示例

  • 使用依赖注入或者配置来动态确定依赖。
// 编译时依赖
class Database {
public:
    void saveData(const std::string& data) {
        // 保存数据到 MySQL 数据库
    }
};

// 运行时依赖
class Database {
public:
    virtual void saveData(const std::string& data) = 0;
};

class MySQLDatabase : public Database {
public:
    void saveData(const std::string& data) override {
        // 保存数据到 MySQL 数据库
    }
};

class PostgresDatabase : public Database {
public:
    void saveData(const std::string& data) override {
        // 保存数据到 PostgreSQL 数据库
    }
};

void saveData(Database* db, const std::string& data) {
    db->saveData(data);  // 根据运行时传入的数据库对象调用相应的方法
}

5. 紧耦合转松耦合

技法说明

  • 减少模块或者类之间的依赖关系,使得各个模块之间可以独立开发、测试和部署。

示例

  • 使用依赖注入、接口抽象等手段降低模块间的耦合度。
// 紧耦合
class ServiceA {
public:
    void doSomething() {
        ServiceB b;
        b.doSomethingElse();
    }
};

class ServiceB {
public:
    void doSomethingElse() {
        // 实现
    }
};

// 松耦合
class ServiceB {
public:
    void doSomethingElse() {
        // 实现
    }
};

class ServiceA {
private:
    ServiceB& b;

public:
    ServiceA(ServiceB& b) : b(b) {}

    void doSomething() {
        b.doSomethingElse();
    }
};

通过这些关键技法,可以更好地进行代码重构,改进现有设计,使得代码更易于维护、扩展和重用,同时提高系统的灵活性和可靠性。

组件协助

定义:现代软件分专业分工之后的第一个结果是框架与应用的区分组建协作模式,通过完绑定来实现框架与应用间的松耦合是二者之间协作时常用的模式。

典型模式
• Template Method
• Observer / Event
• Strategy

动机

在软件构建过程中,每一项任务它常常有稳定的整体操作结构。但各个子步骤却有很多变化的需求。或者由于固定的原因,比如框架与应用之间的关系。而无法和任务的整体结构同时实现。
如何在确定稳定操作结构的前提下,来灵活应对各个子步骤的变化或者晚七实现需求?

结构化软件设计流程

在这里插入图片描述

面向对象软件设计流程

在这里插入图片描述

早绑定与晚绑定

在这里插入图片描述

模式定义

定义一个操作中的算法框架(稳定),而将一些步骤延迟(变化)到子类中template Method 使得子类可以不改变。一个算法的结构即可重新定义就是重写该算法的某些特定步骤。

结构

在这里插入图片描述

要点总结。

模板方法模式是一种非常基础性的设计模式。在面向对象系统中有着大量的应用。他用最简洁的机制也就是虚函数的多态性。为很多应用程序架构提供了灵活的扩展点。是代码复用方面的基本实现结构

除了可以灵活应对自步骤的变化以外,不要调用我,让我来调用你。反向控制结构是模板方法的典型应用。

在具体实验方面,被模板方法调用的虚方法可以具有实现,也可以没有任何实现。也就是所谓的抽象方法和纯虚方法。但一般推荐将他们设置为保护方法

例子

Template Method(模板方法)模式是一种行为设计模式,定义了一个操作中的算法的框架,将一些步骤推迟到子类中实现。这种模式是通过创建一个模板方法,该方法在一个算法的结构中定义了算法的骨架,而将一些步骤的实现延迟到子类中。

以下是一个简单的 C++ 示例,演示了模板方法模式的实现:

#include <iostream>

// 抽象基类
class AbstractClass {
public:
    // 模板方法,定义了算法的骨架
    void templateMethod() {
        operation1();
        commonOperation(); // 模板方法中可以包含具体实现
        operation2();
        if (hookMethod()) { // 钩子方法,可选覆盖
            optionalOperation();
        }
    }

protected:
    // 纯虚函数,需要在子类中实现
    virtual void operation1() = 0;
    virtual void operation2() = 0;

    // 具体方法,子类共享的实现
    void commonOperation() {
        std::cout << "AbstractClass: commonOperation" << std::endl;
    }

    // 钩子方法,可选覆盖
    virtual bool hookMethod() {
        return true; // 默认实现
    }

    // 可选的具体方法,子类可以选择性地覆盖
    virtual void optionalOperation() {
        std::cout << "AbstractClass: optionalOperation" << std::endl;
    }
};

// 具体子类实现
class ConcreteClass1 : public AbstractClass {
protected:
    void operation1() override {
        std::cout << "ConcreteClass1: operation1" << std::endl;
    }

    void operation2() override {
        std::cout << "ConcreteClass1: operation2" << std::endl;
    }

    bool hookMethod() override {
        std::cout << "ConcreteClass1: hookMethod" << std::endl;
        return false; // 覆盖钩子方法
    }
};

// 另一个具体子类
class ConcreteClass2 : public AbstractClass {
protected:
    void operation1() override {
        std::cout << "ConcreteClass2: operation1" << std::endl;
    }

    void operation2() override {
        std::cout << "ConcreteClass2: operation2" << std::endl;
    }

    // 这里没有覆盖钩子方法
};

// 客户端代码
int main() {
    AbstractClass* object1 = new ConcreteClass1();
    AbstractClass* object2 = new ConcreteClass2();

    // 调用模板方法,执行算法
    object1->templateMethod();
    std::cout << std::endl;
    object2->templateMethod();

    delete object1;
    delete object2;

    return 0;
}

示例解释:

  1. AbstractClass 是抽象基类,定义了模板方法 templateMethod() 和一些具体方法 commonOperation()optionalOperation(),以及一个钩子方法 hookMethod()

  2. ConcreteClass1ConcreteClass2 是具体子类,分别实现了抽象类中的纯虚函数 operation1()operation2(),并选择性地覆盖了钩子方法 hookMethod()

  3. 在客户端代码中,创建了 ConcreteClass1ConcreteClass2 的对象,并调用它们的 templateMethod() 方法,执行了定义好的算法。

  4. 输出结果展示了每个步骤的执行顺序,包括了钩子方法的选择性覆盖情况。

这个例子展示了模板方法模式的核心思想:定义一个算法的骨架,将特定的步骤延迟到子类中去实现,从而在不同的子类中实现算法的不同部分,同时保持算法结构的稳定性。

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

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

相关文章

昇思25天学习打卡营第3天|数据集Dataset

一、简介&#xff1a; 数据是深度学习的基础&#xff0c;高质量的数据输入将在整个深度神经网络中起到积极作用。有一种说法是模型最终训练的结果&#xff0c;10%受到算法影响&#xff0c;剩下的90%都是由训练的数据质量决定。&#xff08;doge&#xff09; MindSpore提供基于…

基于STM32的智能病房监控和人脸识别系统设计(毕业设计)

摘 要 随着技术的不断进步和医疗需求的不断增长&#xff0c;智能病房控制系统有望在医疗领域发挥更大的作用。基于此&#xff0c;本文研究设计了一款低成本、操作简单、适用性强的基于STM32的智能病房监控和人脸识别系统。该系统通过STM32作为控制器和OpenMV对人脸分辨进行门…

你好,复变函数1.0

输入时用后缀&#xff0c;开头空格 #include <easyx.h> #include <stdio.h> #define PI 3.141592653589793 #define E 2.718281828459045 #define K (1.0 / 256.0) #define K_1 256.0 //#define LINE//决定函数是用线画还是用点画 struct C {double i;double r;…

同一天里,两位大厂程序员猝死。。。

2024年&#xff0c;真的不是平静的一年。在几天前&#xff0c;IT行业接连发生了两件不幸的事情。 6月17日下午&#xff0c;东南亚电商公司Sh**ee位于北京的研发中心&#xff0c;一位负责研发的女员工突然在工位上晕倒。 同事们赶紧拨打了120&#xff0c;然而还是没能抢救过来&a…

USB2.0网卡安装驱动

有三种安装方式&#xff1a; 驱动精灵驱动总裁USB2.0网卡自带安装程序 前两种很简单&#xff0c;下载驱动精灵或者驱动总裁&#xff0c;然后检测本地硬件&#xff0c;安装相应驱动。 本文重点要介绍的是第三种&#xff0c;利用USB2.0网卡自带的安装程序。有的时候驱动精灵或…

高考志愿填报,如何避免报错专业?

高考志愿填报绝对是关键一环节&#xff0c;分数高低暂且不论&#xff0c;因为这个填报志愿&#xff0c;大概率是决定了余生的职业&#xff0c;也有人说&#xff0c;大学可以转专业&#xff0c;毕业还可以跨行就业&#xff0c;工作了还可以转行.....确实有这个可能性&#xff0c…

如何生成protobuf文件

背景 protobuf是一种用于序列化结构数据的工具&#xff0c;实现数据的存储与交换&#xff0c;与编程语言和开发平台无关。 序列化&#xff1a;将结构数据或者对象转换成能够用于存储和传输的格式。 反序列化&#xff1a;在其他的计算环境中&#xff0c;将序列化后的数据还原为…

【React】AntD组件---极客园--01.项目前置准备

项目搭建 基于CRA创建项目 CRA是一个底层基于webpack快速创建React项目的脚手架工具 # 使用npx创建项目 npx create-react-app react-jike# 进入到项 cd react-jike# 启动项目 npm start调整项目目录结构 -src-apis 项目接口函数-assets 项目资源文件&…

【鸿蒙】创建第⼀个鸿蒙项⽬

点击 Create Project 配置项目 开发工具界面 工程介绍

工业边缘计算网关

1 介绍 HINETG系列边缘计算网关&#xff08;Linux操作系统&#xff09;&#xff0c;是华辰智通的—款面向工业现场设备接入、数据采集、设备监控的工业级边缘计算网关。采用ARM Cortex-A7 800MHz高性能CPU,拥有以太网、串口、CAN口、IO口等丰富的接口&#xff0c;支持以太网、…

docker基础使用教程

1.准备工作 例子&#xff1a;工程在docker_test 生成requirements.txt文件命令&#xff1a;&#xff08;使用参考链接2&#xff09; pip list --formatfreeze > requirements.txt 参考链接1&#xff1a; 安装pipreqs可能比较困难 python 项目自动生成环境配置文件require…

教程:LVM操作讲解

LVM简介 在系统运维过程中&#xff0c;对磁盘扩缩容是常见的操作。如何高效的管理磁盘容量&#xff0c;lvm提供了很好的解决方案。 LVM将磁盘抽象成PV、VG、LV&#xff0c;方便用户进行磁盘管理&#xff0c;简单来讲&#xff0c;是由物理磁盘划分成PV&#xff0c;PV加入到具体…

Ubuntu 22.04.4 LTS openresty(Nginx) 通过Lua+Redis 实现动态封禁IP

1 系统环境 testiZbp1g7fmjea77vsqc5hmmZ:~$ cat /etc/os-release PRETTY_NAME"Ubuntu 22.04.4 LTS" NAME"Ubuntu" VERSION_ID"22.04" VERSION"22.04.4 LTS (Jammy Jellyfish)" VERSION_CODENAMEjammy IDubuntu ID_LIKEdebian HOME…

蓝桥杯 经典算法题 实现归并排序

题目&#xff1a; 题解&#xff1a; 不断地将数组不断向下平均分为两部分&#xff0c;直到每个子数组中元素数量为1&#xff0c;这样就可以将相邻两个数组长度为1的数组看作是单调数组合并为一个大的单调数组&#xff0c;如此不断向上合并出最终的单调数组。 #include <bi…

Golang | Leetcode Golang题解之第173题二叉搜索树迭代器

题目&#xff1a; 题解&#xff1a; type BSTIterator struct {stack []*TreeNodecur *TreeNode }func Constructor(root *TreeNode) BSTIterator {return BSTIterator{cur: root} }func (it *BSTIterator) Next() int {for node : it.cur; node ! nil; node node.Left {it…

jemeter基本使用

后端关验签&#xff0c;设置请求头编码和token 配置编码和token

用全志T113做了块多功能卡片电脑,成本只要60块

FunnyPi-T113是一款基于全志T113-S3/D1S处理器的完全开源多功能开发板&#xff0c;设计FunnyPi最初的目的是想借此T113卡片电脑来满足日常学习&#xff0c;并结合T113高效能和低功耗的特点&#xff0c;来满足像语音助手&#xff0c;智能家居屏幕、桌面摆件屏、博客服务器等嵌入…

激励-保健理论和公平理论

激励-保健理论 herzberg的激励-保健理论中&#xff0c;保健因素是context of a job&#xff0c;激励因素是content of a job。 context of a job是受组织控制的因素&#xff0c;比如工作条件&#xff0c;基本工资&#xff0c;公司政策等&#xff0c;个人无法支配。content of…

R语言——绘图与数据可视化

1、练习将25个点的符号绘制出来&#xff0c;然后用rainbow()返回25个颜色&#xff0c;后5个符号形状的背景颜色用蓝色填充&#xff0c;图的标题为"符号图"&#xff0c;x轴标题为符号索引&#xff0c;y轴标题为符号形状。 2、根据员工的销售业绩画饼状图&#xff0c;添…

Vitis Accelerated Libraries 学习笔记--OpenCV 运行测试

目录 1. 简介 2. 实例测试 2.1 实例介绍 2.1 创建工程 3 常见错误 3.1 核心共享库报错 4. 总结 1. 简介 在《Vitis Accelerated Libraries 学习笔记--OpenCV 安装指南-CSDN博客》一文中&#xff0c;我详尽地介绍了 OpenCV 的安装过程。尽管 Vitis Vision 库的实现本身并…