设计模式4-模版方法

设计模式

  • 重构获得模式
  • 重构的关键技法
      • 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.mzph.cn/web/32846.shtml

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

相关文章

[SAP ABAP] 排序内表数据

语法格式 整表排序 SORT <itab> [ASCENDING|DESCENDING]. 按指定字段排序 SORT <itab> BY f1 [ASCENDING|DESCENDING] f2 [ASCENDING|DESCENDING] ... fn [ASCENDING|DESCENDING].<itab>&#xff1a;代表内表 不指定排序方式则默认升序排序 示例1 结果显…

.hmallox勒索病毒解密方法|勒索病毒解决|勒索病毒恢复|数据库修复

导言&#xff1a; 在当今数字化时代&#xff0c;勒索病毒已经成为网络安全的一大威胁&#xff0c;其中包括了最近出现的.hmallox勒索病毒。这类恶意软件不仅能够对计算机系统进行加密&#xff0c;还会要求用户支付赎金以换取解密密钥&#xff0c;给个人用户和企业带来了严重的…

C++的特殊类设计 饥饿汉模式

目录 特殊类设计 设计一个不能被拷贝的类 设计一个只能在堆上创建对象的类 设计一个只能在栈上创建对象的类 设计一个不能继承的类 设计模式 单例模式 饿汉模式 饥汉模式 特殊类设计 设计一个不能被拷贝的类 C98的设计方式&#xff1a;将该类的拷贝构造和赋值运算符…

一小时搞定Git(含盖IDEA使用)

文章目录 1. git基本概念1.1版本控制1.1.1 版本控制软件 2. 命令的使用2.1 Linux命令2.2 git基础指令2.2.1 设置用户2.2.2 初始化本地仓库2.2.3 查看本地仓库状态2.2.4 添加暂存区域2.2.5 提交本地库2.2.6 切换版本 2.3 分支操作2.3.1 分支基本操作2.3.2 合并操作2.3.4 分支开发…

基于STM32的智能家居安防系统

目录 引言环境准备智能家居安防系统基础代码实现&#xff1a;实现智能家居安防系统 4.1 数据采集模块4.2 数据处理与分析4.3 控制系统实现4.4 用户界面与数据可视化应用场景&#xff1a;智能家居安防管理与优化问题解决方案与优化收尾与总结 1. 引言 智能家居安防系统通过使…

195.回溯算法:分割回文串(力扣)

代码解决 class Solution { public:vector<string> res; // 当前路径&#xff0c;用于存储一个可能的回文分割结果vector<vector<string>> result; // 存储所有可能的回文分割结果// 判断子串 s[left:right] 是否是回文bool isPalindrome(const string& …

Linux应用系统快速部署:docker快速部署linux应用程序

目录 一、背景 &#xff08;一&#xff09;引入docker的起因 &#xff08;二&#xff09;docker介绍 &#xff08;三&#xff09;Docker部署的优势 1、轻量级和可移植性 2、快速部署和扩展 3、一致性 4、版本控制 5、安全性 6、资源隔离 7、简化团队协作 8、多容器…

VMware虚拟机三种网络模式设置 - Bridged(桥接模式)

一、前言 由于linux目前很热门&#xff0c;越来越多的人在学习linux&#xff0c;但是买一台服务放家里来学习&#xff0c;实在是很浪费。那么如何解决这个问题&#xff1f;虚拟机软件是很好的选择&#xff0c;常用的虚拟机软件有vmware workstations和virtual box等。 在使用虚…

基于imx6ull开发板 移植opencv4.7.0

一、概述 本章节是针对opencv-4.7.0移植到Linux系统&#xff0c;运行在正点原子-I.MX6U ALPHA开发板 上&#xff0c;详细的移植流程如下。 二、环境要求 2.1 硬件环境 正点原子-I.MX6U ALPHA开发板虚拟机&#xff1a;VMware 2.2 软件环境 Ubuntu系统要求&#xff1a;20.0…

DIVE INTO DEEP LEARNING 50-55

文章目录 50. semantic segmentation50.1 Basic concepts50.2 Major application 51. Transposed convolution51.1 Basic concepts51.2 Major role51.3 Implementation steps and application areas51.4 Transposed convolution51.5 Transposed convolution is a type of convo…

谁说串口通信波特率越高越好?

在电子世界里&#xff0c;串口通信就像是电子设备之间的“悄悄话”&#xff0c;它们通过串行数据传输来交换信息。但你知道吗&#xff1f;串口通信的波特率并不是越高越好&#xff0c;这事儿得好好聊聊。 1.什么是串口通信&#xff1f; 串口通信&#xff0c;就像它的名字一样&a…

深度学习11-13

1.神经元的个数对结果的影响&#xff1a; &#xff08;http://cs.stanford.edu/people/karpathy/convnetjs/demo/classify2d.html&#xff09; &#xff08;1&#xff09;神经元3个的时候 &#xff08;2&#xff09;神经元是10个的时候 神经元个数越多&#xff0c;可能会产生…

计算机组成原理 —— 存储系统(DRAM和SRAM,ROM)

计算机组成原理 —— 存储系统&#xff08;DRAM和SRAM&#xff09; DRAM和SRAMDRAM的刷新DRAM地址复用ROM&#xff08;Read-Only Memory&#xff08;只读存储器&#xff09;&#xff09; 我们今天来看DRAM和SRAM&#xff1a; DRAM和SRAM DRAM&#xff08;动态随机存取存储器&…

展示3D模型的网站哪个好?

如果仅仅是模型展示&#xff0c;目前国内外值得推荐的无非就是那么几个&#xff0c;它们各自有不同的特点和优势&#xff1a; 1、Sketchfab&#xff1a;Sketchfab是一个知名的3D模型展示平台&#xff0c;提供了海量的模型资源和出色的3D展示效果。用户无需安装任何插件即可在线…

Spring Cloud - nacos +ubuntu环境搭建

1、安装ubuntu虚拟环境 VMware虚拟机安装Ubuntu与配置Ubuntu&#xff08;超详细教程&#xff09; 2、docker环境安装 1、apt-get install apt-transport-https ca-certificates curl gnupg-agent software-properties-common 2、curl -fsSL https://mirrors.ustc.edu.cn/docke…

Maven的依赖传递、依赖管理、依赖作用域

在Maven项目中通常会引入大量依赖&#xff0c;但依赖管理不当&#xff0c;会造成版本混乱冲突或者目标包臃肿。因此&#xff0c;我们以SpringBoot为例&#xff0c;从三方面探索依赖的使用规则。 1、 依赖传递 依赖是会传递的&#xff0c;依赖的依赖也会连带引入。例如在项目中…

自动驾驶仿真测试用例表格示例 ACC ELK FCW

自动驾驶仿真测试用例表格示例 测试用例概览 本测试用例表格涵盖了自动驾驶系统中多个关键功能和场景的测试&#xff0c;旨在确保系统在不同条件下的表现和稳定性。 用例编号测试项目测试描述预期结果实际结果通过/失败TC-001ACC功能测试在高速公路上启用ACC&#xff0c;测试车…

SCI一区级 | Matlab实现BO-Transformer-LSTM多变量时间序列预测

SCI一区级 | Matlab实现BO-Transformer-LSTM多变量时间序列预测 目录 SCI一区级 | Matlab实现BO-Transformer-LSTM多变量时间序列预测效果一览基本介绍程序设计参考资料 效果一览 基本介绍 1.【SCI一区级】Matlab实现BO-Transformer-LSTM多变量时间序列预测&#xff0c;贝叶斯…

力扣SQL50 超过5名学生的课

Problem: 596. 超过5名学生的课 Code select class from courses group by class having count(distinct student) > 5;

高通安卓12-固件升级

下载步骤 第一步 格式化 「下载一次即可&#xff1b;能开机能下载的板子 忽略这一步&#xff0c;直接执行第二步即可」 QFIL工具配置为UFS类型&#xff0c;勾选Provision&#xff0c;如下图&#xff1a; Programmer选择prog_firehose_ddr.elf&#xff0c;Provision Xml选择prov…