设计模式——模版方法和策略模式

news2024/9/21 2:35:26
在这里插入图片描述

前言

作为一名资深CV工程师,学会为自己减少工作量乃重中之重。但只是一味地CV,只会因为劣质代码而让自己的工作量加倍,为了将来不被繁重的维护工作而打扰自己的休息日,为了更好的节能,学习设计模式,刻不容缓。

模版方法

概念

生活中我们总是离不开各种模版的存在,作文、文章、简历的模版。正是因为这些模板的存在,我们工作的效率才大大提高,而设计模式的中模版方法正是与现实生活中的模版如出一辙。

现实中的模版一般有两类,一类为只有大体框架的,而另一类为全部都填写完的,只需要修改你想要的部分即可。我们在设计模版方法的时候,可以设计一个骨架作为调用流程,而具体的功能/算法则留给派生类来实现。或者,将那些会变化的代码逻辑封装起来,如果有需要再留给派生类更改。

定义

模版方法指的是为算法定义一个大体运行框架,而将那些会变化的代码封装起来,而留给派生类实现的一种设计模式。

代码实现

以冲泡饮料为例,冲泡的步骤基本都不会改变,只有一部分细节会有变化,正好可以使用模版方法进行设计封装。

Beverage
+void PrepareRecipe()
+~Beverage()
#void BoilWater()
#void PourInCup()
#void Brew()
#void AddCondiments()
Coffee
#void Brew()
#void AddCondiments()
Tea
#void Brew()
#void AddCondiments()

代码实例

class Beverage
{
public:
    void PrepareRecipe()
    {   // 代码运行的大体框架
        BoilWater();
        PourInCup();
        Brew();
        AddCondiments();
    }
    virtual ~Beverage() = default;
protected:
    // 具体实现可留给子类实现
    virtual void BoilWater()
    {
        std::cout << "Boiling water" << std::endl;
    }

    virtual void PourInCup()
    {
        std::cout << "Pouring into cup" << std::endl;
    }

    virtual void Brew() = 0;

    virtual void AddCondiments() = 0;
};

class Coffee : public Beverage
{
protected:
    void Brew() override { std::cout << "Brewing coffee" << std::endl; }
    void AddCondiments() override { std::cout << "Adding sugar and milk" << std::endl; }
};

class Tea : public Beverage
{
protected:
    void Brew() override { std::cout << "Brewing tea" << std::endl; }
    void AddCondiments() override { std::cout << "Adding lemon" << std::endl; }
};

钩子方法

钩子方法是模版方法的一种变体,它在框架中定义一个判断方法(钩子),让子类来决定其代码逻辑,减少了减少了外部的干预,提高了代码的灵活度与拓展性。

继续用上方的饮料代码来举例就是,不是所有都饮料中都需要添加调味剂/配料,此时可以由子类决定其算法逻辑。

代码实例

class Beverage
{
public:
    void PrepareRecipe()
    {   // 代码运行的大体框架
        BoilWater();
        PourInCup();
        Brew();
        if (NeedCondiments()) AddCondiments();
    }
    virtual ~Beverage() = default;
protected:
    // 钩子方法
    virtual bool NeedCondiments() { return false; }
    // 具体实现可留给子类实现
    virtual void BoilWater()
    {
        std::cout << "Boiling water" << std::endl;
    }

    virtual void PourInCup()
    {
        std::cout << "Pouring into cup" << std::endl;
    }

    virtual void Brew() {std::cout << "Brew some drink"; };

    virtual void AddCondiments() { std::cout << "Adding some condiments"; }
};

class Coffee : public Beverage
{
protected:
    bool NeedCondiments() override { return true; }
    void Brew() override { std::cout << "Brewing coffee" << std::endl; }
    void AddCondiments() override { std::cout << "Adding sugar and milk" << std::endl; }
};

策略模式

概念

策略模式指的是将算法封装起来(成员变量/接口),使其能够根据不同情况而更换。策略模式与模版方法都需要将其算法/实现封装起来,初认可能会将其混淆,但只要认清模版方法是将实现延迟到子类实现,而策略模式是变化封装成类(接口/委托),就不会混淆了。

这里使用支付系统作为例子,随着互联网的发展,我们的支付方式越发变得丰富,如果每增加一个支付方式,支付系统都要重写代码的话,那么想必程序员都再也不用担心失业了。这种情况下使用策略模式,将支付手段封装起来,那么就正好符合OO原则中的开闭原则,系统的维护性也更好。

Payment
+Pay(int amount)
CreditCardPayment
+Pay(int amount)
PayPalPayment
+Pay(int amount)
PaymentContext
-Payment* _payment
+PayAmount(int amount)
+SetPayment(Payment* payment)

实现

class Payment
{
public:
    virtual void Pay(int amount) = 0; // 抽象支付方式
    virtual ~Payment() = default;
};

// 不同支付方式继承与同一个接口
class CreditCardPayment : public Payment
{
public:
    void Pay(int amount) override
    {
        std::cout << "Paying " << amount << " using CreditCard" << std::endl;
    }
};

class PayPalPayment : public Payment
{
public:
    void Pay(int amount) override
    {
        std::cout << "Paying " << amount << " using PayPal" << std::endl;
    }
};

class PaymentContext
{
private:
    Payment* _payment{};
public:
    void PayAmount(int amount)
    {
        if (_payment) _payment->Pay(amount);
    }
    // 根据需要变更策略
    void SetPayment(Payment* payment) { _payment = payment;}
};

总结

特点模板方法模式策略模式
定义将算法的固定部分提取到基类,变化部分由子类实现。将不同算法封装成独立的类,通过上下文类动态切换算法。
设计意图通过基类定义算法框架,将具体实现延迟到子类。通过将算法封装成独立的类,使其能够在运行时动态替换。
使用场景固定流程的多个步骤,其中部分步骤的实现因子类不同而不同。多种算法可以互换,且算法相对独立,变化频繁。
优点1. 代码复用性高。
2. 易于扩展新功能。
1. 符合开闭原则。
2. 代码更加灵活,易于维护和扩展。
缺点1. 继承关系较复杂。
2. 增加类的数量。
1. 增加系统复杂度。
2. 上下文类需要了解所有策略的细节。

📜博客主页:主页
📫我的专栏:C++
📱我的github:github

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

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

相关文章

goLang 通过 Select + Channels 实现定时任务与调度

time 包 在与 channel 结合时提供了很多有用的功能,其中 time.Ticker 结构体能够处理定时事件,它会定期在指定 channel 上发送时间值。在这篇文章中,我们深入探讨了如何使用 Go 语言中的 time 包与 channel 结合来处理定时事件和任务调度。通过 time.Ticker 结构体的周期性触…

MySQL(终章)视图, 用户管理, C连接MySQL.

目录 1.视图; 2.用户管理; 1.视图: 1.1 概念和基本操作: 视图本质就是表结构, 虚拟表. 视图和基表数据的改变都会相互影响. 创建视图语法: create view 视图名 as select语句&#xff1b; 修改视图: update set ; 删除视图: drop view 视图名&#xff1b; 1. 2 视图使用规则: …

【java】力扣 跳跃游戏

文章目录 题目链接题目描述代码1.动态规划2.贪心 题目链接 55.跳跃游戏 题目描述 代码 1.动态规划 1.1 dp数组的含义 dp[i]&#xff1a;从[0,i]的任意一点处出发&#xff0c;你最大可以跳跃到的位置。 例如nums[2,3,1,1,4]中: dp[0]2 dp[1]4 dp[2]4 dp[3]4 dp[4]8&#xff…

PACS-医学影像信息管理系统,全影像科室PACS源码,内置包括MPR、CMPR、VR等三维处理功能

PACS系统可以覆盖医院现有放射、CT、MR、核医学、超声、内镜、病理、心电等绝大部分DICOM和非DICOM检查设备&#xff0c;支持从科室级、全院机、集团医院级乃至到区域PACS的平滑扩展&#xff0c;能够与医院HIS、集成平台的有效集成和融合&#xff0c;帮助医院实现了全院医学影像…

大数据-38 Redis 高并发下的分布式缓存 Redis简介 缓存场景 读写模式 旁路模式 穿透模式 缓存模式 基本概念等

点一下关注吧&#xff01;&#xff01;&#xff01;非常感谢&#xff01;&#xff01;持续更新&#xff01;&#xff01;&#xff01; 目前已经更新到了&#xff1a; HadoopHDFSMapReduceHiveFlumeSqoopZookeeperHBaseRedis 章节内容 上一节我们完成了&#xff1a; HBase …

更改若依框架的包名

使用KIT更改 官网&#xff1a;http://www.goldpankit.com 具体步骤 1. 打开KIT官网&#xff0c;选择Ruoyi-Vue框架&#xff0c;如下图所示 2. 点击构建框架按钮&#xff0c;如下图所示 3. 填写框架参数并点击立即构建 4. 下载ZIP并解压即可&#xff0c;构建后内容如下 这样…

《Exploring Aligned Complementary Image Pair for Blind Motion Deblurring》

这篇论文的标题《Exploring Aligned Complementary Image Pair for Blind Motion Deblurring》可以翻译为《探索对齐的互补图像对用于盲运动去模糊》。从标题可以推断,论文的焦点在于开发一种算法或技术,利用成对的图像来解决运动模糊问题,特别是在不知道模糊核(即造成模糊…

基于牛顿-拉夫逊优化算法(Newton-Raphson-based optimizer, NBRO)的无人机三维路径规划

牛顿-拉夫逊优化算法(Newton-Raphson-based optimizer, NBRO)是一种新型的元启发式算法&#xff08;智能优化算法&#xff09;&#xff0c;该成果由Sowmya等人于2024年2月发表在中科院2区Top SCI期刊《Engineering Applications of Artificial Intelligence》上。 1、算法原理…

(vue)Vue读取public中的json文件,打包后只需更改包文件

(vue)Vue读取public中的json文件,打包后只需更改包文件 背景&#xff1a;增加账号需求。原本是在页面&#xff0c;每次都需技术人员添加再打包部署&#xff0c;现在放到json里&#xff0c;以后直接服务器改json就行。 旧版&#xff1a; let userArr [{username:aaa,password:…

系统架构设计师教程 第3章 信息系统基础知识-3.7 企业资源规划(ERP)-解读

系统架构设计师教程 第3章 信息系统基础知识-3.7 企业资源规划&#xff08;ERP&#xff09; 3.7.1 企业资源规划的概念3.7.2 企业资源规划的结构3.7.2.1 生产预测3.7.2.2 销售管理&#xff08;计划&#xff09;3.7.2.3 经营计划&#xff08;生产计划大纲&#xff09;3.7.2.4 …

IDEA双击卡住打不开问题解决

双击idea图标一直卡在这个页面 解决方案: 随便找个文件 edit with intellij idea打开 删除最近安装的插件 apply 然后重启

C++容器之array

C array容器 特点 array是固定大小数组。 C中&#xff0c;array是一个模板类&#xff0c;提供了数组的功能&#xff0c;在许多方面比原始的 C 风格数组更为强大和灵活。 主要特点 固定大小: std::array 的大小在编译时确定&#xff0c;运行时无法修改。包含全部 STL 功能:…

神经网络之循环神经网络

目录 一、循环神经网络概述&#xff1a;1.传统神经网络与循环神经网络的区别&#xff1a;2.循环神经网络定义&#xff1a; 图片来自&#xff1a;深度学习———循环神经网络 一、循环神经网络概述&#xff1a; 1.传统神经网络与循环神经网络的区别&#xff1a; MLP、卷积神经…

【单目3D检测】smoke(1):模型方案详解

纵目发表的这篇单目3D目标检测论文不同于以往用2D预选框建立3D信息&#xff0c;而是采取直接回归3D信息&#xff0c;这种思路简单又高效&#xff0c;并不需要复杂的前后处理&#xff0c;而且是一种one stage方法&#xff0c;对于实际业务部署也很友好。 题目&#xff1a;SMOKE&…

【LabVIEW作业篇 - 4】:属性节点赋值和直接节点赋值的区别体现

文章目录 属性节点赋值和直接节点赋值的区别体现 属性节点赋值和直接节点赋值的区别体现 创建5个圆形指示灯&#xff0c;然后循环点亮&#xff0c;先给圆形指示灯赋值假变量&#xff0c;然后再进行循环。 运行结果&#xff0c;观察结果&#xff0c;发现刚开始运行时&#xff0…

Unity | Shader基础知识(第十九集:顶点着色器的进一步理解-易错点讲解)

目录 一、前言 二、网格 三、方法UnityObjectToClipPos 四、顶点着色器和片元着色器的POSITION 五、作者的碎碎念 一、前言 之前我们简单讲解过顶点着色器&#xff0c;也简单讲解了表面着色器&#xff0c;并且一起做了一些案例&#xff0c;因为顶点着色器本身是更自由一些…

云计算实训09——rsync远程同步、自动化推取文件、对rsyncd服务进行加密操作、远程监控脚本

一、rsync远程同步 1.rsync基本概述 &#xff08;1&#xff09;sync同步 &#xff08;2&#xff09;async异步 &#xff08;3&#xff09;rsync远程同步 2.rsync的特点 可以镜像保存整个目录树和文件系统 可以保留原有权限&#xff0c;owner,group,时间&#xff0c;软硬链…

IDEA的工程与模块管理

《IDEA破解、配置、使用技巧与实战教程》系列文章目录 第一章 IDEA破解与HelloWorld的实战编写 第二章 IDEA的详细设置 第三章 IDEA的工程与模块管理 第四章 IDEA的常见代码模板的使用 第五章 IDEA中常用的快捷键 第六章 IDEA的断点调试&#xff08;Debug&#xff09; 第七章 …

Harbor系列之1:介绍、架构及工作流程说明

Harbor介绍、架构及工作流程说明 Harbor 是一个用于存储、签名和扫描内容的企业级容器镜像注册表项目。由 VMware 开发并于 2016 年开源。Harbor 提供了一些关键特性&#xff0c;使其成为企业使用的理想选择。 1. Harbor 介绍 1.1 什么是 Harbor Harbor 是一个开源的云原生…

Python酷库之旅-第三方库Pandas(034)

目录 一、用法精讲 101、pandas.Series.__array__魔法方法 101-1、语法 101-2、参数 101-3、功能 101-4、返回值 101-5、说明 101-6、用法 101-6-1、数据准备 101-6-2、代码示例 101-6-3、结果输出 102、pandas.Series.get方法 102-1、语法 102-2、参数 102-3、…