23种设计模式之C++实践

news2025/1/22 18:58:49

23种设计模式之C++实践

  • 1. 简介
  • 2. 基础知识
  • 3. 设计模式
    • (一)创建型模式
      • 1. 单例模式
        • 1.2 饿汉式单例模式
        • 1.3 懒汉式单例模式
        • 比较
        • IoDH
        • 单例模式总结
      • 2. 简单工厂模式
        • 简单工厂模式总结
      • 3. 工厂方法模式
        • 工厂方法模式总结
      • 4. 抽象工厂模式
        • 抽象工厂模式总结
      • 5. 原型模式
        • 原型模式总结
      • 6. 建造者模式
        • 建造者模式总结

1. 简介

设计模式是一门技术,更是一门艺术,它为构建可维护性和可复用性俱佳的软件而诞生。

2. 基础知识

  1. 设计模式(Design Pattern):是一套被反复使用的,多数人知晓的,经过分类编目的代码设计经验的总结,使用设计模式是为了可以重用代码,让代码更容易被理解提高代码的可靠性
  2. UML(United Modeling Language)
      1. 关联关系(实线):通常将一个类的对象作为另一个类的成员变量。
        1. 双向关联
        2. 单向关联
        3. 自关联
        4. 多重性关联
      2. 聚合关系(空心菱形直线):成员是整体的一部分,但是成员可以脱离整体存在。通常通过set()函数初始化成员变量。
      3. 组合关系(实心菱形直线):整体控制成员的声明周期,整体不存在,则成员不存在。通常通过构造函数初始化成员变量。
      4. 依赖关系(带箭头虚线):类的改变影响到使用该类的其他类,则其他类依赖该类。通常通过将一个类的对象作为另一个类中方法的参数体现。
      5. 泛化关系/继承关系(空心三角形实线):父类与子类。
      6. 接口与实现关系(空心三角形虚线):在接口类中声明抽象函数,在实现类中实现函数。
  3. 面向对象设计原则
    1. 单一职责原则(Single Responsibility):一个类只负责一个功能领域中的相应职责。或者可以定义为:就一个类而言,应该只有一个引起它变化的原因。高内聚,低耦合
    2. 开闭原则(Open-Closed Principle,OCP):一个软件实体应该对外扩展开放,对修改关闭。即软件实体尽量在不修改原有代码的基础下进行扩展。抽象化
    3. 里氏代换原则(Liskov Substitution Principle,LSP):所有引用父类的地方必须能透明地使用其子类的对象。父类抽象化
    4. 依赖倒转原则(Dependency Inversion Principle,DIP):抽象不应该依赖于细节,细节应该依赖于抽象。即针对接口编程,而不是针对实现编程。参数抽象化
    5. 接口隔离原则(Interface Segregation Principle,ISP):使用多个专用的接口,而不适用单一的总接口,即客户端不应该依赖那些它不需要的接口。
    6. 合成复用原则(Composite Reuse Principle,CRP):尽量使用对象组合,而不是继承来达到复用的目的。
    7. 迪米特法则/最少知识原则(Law of Demeter/Least Knowledge Principle,LoD/LKP):一个软件实体应该尽可能少的与其他实体发生相互作用。

3. 设计模式

(一)创建型模式

1. 单例模式

  1. 单例模式(Singleton Pattern):确保某一个类只有一个实例,而且自行实例化并向整个系统提供这个实例,这个类成为单例类,它提供全局访问的方法。

  2. 要点:

    1. 某个类只能有一个实例

    2. 它必须自行创建这个实例

    3. 它必须自行向整个系统提供这个实例

  3. 结构图:

  4. 使用场景实例

    Windows任务管理器

    在一个Windows系统中,任务管理器存在唯一性。

  5. 代码实例

    // TaskManager.h
    class TaskManager {
         
     private:
      /**
       * @brief 初始化窗口
       *
       */
      TaskManager() {
         }
    
     public:
      /**
       * @brief 显示进程
          *
          */
    
        void displayProcesses();
    
      /**
       * @brief 显示服务
          *
          */
    
        void displayServices();
    
     private:
      static TaskManager* tm;
    
     public:
      static TaskManager* getInstance();
    };
    
    // TaskManager.cpp
    TaskManager* TaskManager::tm = nullptr;
    
    void TaskManager::displayProcesses() {
         
      printf("显示进程……\n");
      return;
    }
    void TaskManager::displayServices() {
         
      printf("显示服务……\n");
      return;
    }
    
    TaskManager* TaskManager::getInstance() {
         
      if (tm == nullptr) {
         
        tm = new TaskManager();
      }
      return tm;
    }
    
  6. 代码测试

    • 测试代码:

      int main(int argc, char** argv) {
             
        printf("I'm Singleton Pattern!\n");
        // begin test
        SingleTonNS::TaskManager* tm = SingleTonNS::TaskManager::getInstance();
        tm->displayProcesses();
        tm->displayServices();
        // end test
        return 0;
      }
      
    • 输出

      I’m Singleton Pattern!
      显示进程……
      显示服务……

1.2 饿汉式单例模式
  1. 饿汉式单例模式(Eager Singleton):在定义静态变量的时候实例化单例类,因此在类加载时就已经创建了单例对象

  2. 结构图

  3. 代码示例

     // TaskManager.h
     class TaskManager {
         
      private:
       /**
        * @brief 初始化窗口
        *
        */
       TaskManager() {
         }
    
      public:
       /**
        * @brief 显示进程
           *
           */
    
         void displayProcesses();
    
       /**
        * @brief 显示服务
           *
           */
    
         void displayServices();
    
      private:
       static TaskManager* tm;
    
      public:
       static TaskManager* getInstance();
     };
    
     // TaskManager.cpp
     // 饿汉式单例模式初始化
     TaskManager* TaskManager::tm = new TaskManager();
    
     void TaskManager::displayProcesses() {
         
       printf("显示进程……\n");
       return;
     }
     void TaskManager::displayServices() {
         
       printf("显示服务……\n");
       return;
     }
     TaskManager* TaskManager::getInstance() {
         
       return tm;
     }
    
1.3 懒汉式单例模式
  1. 见单例模式实现方式。
比较
  1. 饿汉式单例类

    • 优点

      1. 无需考虑多线程访问问题,可以确保实例的唯一性

      2. 调用速度与反应时间更快

    • 缺点

      1. 资源利用效率更低

      2. 加载时间更长

  2. 懒汉式单例类

    • 优点
      1. 延迟加载,无需一直占用系统资源
    • 缺点
      1. 必须处理好多线程同时访问的问题,可能引起性能受影响
IoDH
  1. 有没有一种方法,能够将两种单例的缺点都克服,而将两者的优点合二为一呢?答案是肯定的,即Initialization on Demand Holder技术
  2. IoDH:在单例类中增加一个静态内部类,在该内部类中,创建单例对象,再将该单例对象通过getInstance()方法返回给外部使用。
  3. 使用IoDH,既可以实现延迟加载,又可以保证线程安全,不影响系统性能,但是与编程语言本身的特性相关,很多面向对象语言不支持IoDH,例如C++。支持的语言有Java。
单例模式总结
  • 优点
    1. 提供了对唯一实例的受控访问。
    2. 节约系统资源
    3. 允许可变数目的实例。
  • 缺点
    1. 没有抽象层,扩展较为困难
    2. 职责过重,一定程度上违反了单一职责原则。
    3. 很多面向对象语言(例如Java,C#)的运行环境都提供了自动垃圾回收技术,因此,如果实例化的共享对象长期不被利用,会自动销毁并回收资源,下次利用时重新实例化,这将导致共享的单例对象状态的丢失。
  • 适用场景
    1. 系统只需要一个实例对象
    2. 客户调用类的单个实例只允许使用一个公共访问点。

2. 简单工厂模式

  1. 简单工厂模式(Simple Factory Pattern):定义一个工厂类,它可以根据参数的不同返回不同类的实例,被创建类的实例通常具有共同的父类。

  2. 要点:

    1. 当你需要什么,只需要传入一个正确的参数,就可以获取你所需要的对象,而无须知道其创建细节。
  3. 结构图:

  4. 适用场景实例

    图标库:为应用系统提供各种不同外观的图表,例如柱状图、饼状图、折线图等等

  5. 代码示例

// Chart.h
/**
 * @brief 图表类接口
 *
 */
class Chart {
   
 public:
  virtual void display() = 0;
};

/**
 * @brief 柱状图
 *
 */
class HistogramChart : public Chart {
   
 public:
  HistogramChart();
  void display() override;
};

/**
 * @brief 饼状图
 *
 */
class PieChart : public Chart {
   
 public:
  PieChart();
  void display() override;
};

/**
 * @brief 折线图
 *
 */
class LineChart : public Chart {
   
 public:
  LineChart();
  void display() override;
};

// ChartFactory.h
class ChartFactory {
   
 public:
  static Chart* getChart(std::string type);
};

// Chart.cpp
HistogramChart::HistogramChart() {
    printf("创建柱状图!\n"); }

void HistogramChart::display() {
   
  printf("展示柱状图!\n");
  return;
}

PieChart::PieChart() {
    printf("创建饼状图!\n"); }

void PieChart::display(

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

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

相关文章

MySQL索引优化实战一

#插入一些示例数据drop procedure if exists insert_emp;delimiter ;;create procedure insert_emp()begindeclare i int;set i1;while(i<100000)doinsert into employees(name,age,position) values(CONCAT(tqq,i),i,dev);set ii1;end while;end;;delimiter ;call insert_e…

linaro交叉编译工具链下载与使用笔记

笔记 文章目录 笔记确定目标 &#xff08;aarch64&#xff09;选择版本&#xff08;7.5&#xff09;选择目标&#xff08;aarch64-linux-gnu&#xff09;下载地址工具链&#xff08;gcc-linaro-7.5.0-2019.12-x86_64_aarch64-linux-gnu.tar.xz&#xff09;编译测试 &#xff08…

WIFI模块(esp-01s)实现天气预报代码实现

目录 前言 实现图片 一、串口编程的实现 二、发送AT指令 esp01s.c esp01s.h 三、数据处理 1、初始化 2、cjson处理函数 3、核心控制代码 四、修改堆栈大小 前言 实现图片 前面讲解了使用AT指令获取天气与cjson的解析数据&#xff0c;本章综合将时间显示到屏幕 一、…

Python超级详细的变量命名规则

Python 需要使用标识符给变量命名&#xff0c;其实标识符就是用于给程序中变量、类、方法命名的符号&#xff08;简单来说&#xff0c;标识符就是合法的名字&#xff09;。 Python 语言的标识符必须以字母、下画线&#xff08;_&#xff09;开头&#xff0c;后面可以跟任意数目…

Python单元测试之道:从入门到精通的全面指南

在这篇文章中&#xff0c;我们会深入探讨Python单元测试的各个方面&#xff0c;包括它的基本概念、基础知识、实践方法、高级话题&#xff0c;如何在实际项目中进行单元测试&#xff0c;单元测试的最佳实践&#xff0c;以及一些有用的工具和资源 一、单元测试重要性 测试是软…

ELk部署,保姆级教学超详细!!!

Elk&#xff08;Elasticsearch, Logstash, Kibana&#xff09;是一套日志收集、存储和展示方案&#xff0c;是由Elastic公司开发的开源软件组合。 Elasticsearch&#xff1a;是一个分布式的搜索和分析引擎。它能够处理大量的数据&#xff0c;并提供快速、准确的搜索结果&#x…

在线 SQL 模拟器SQL Fiddle使用简介

在线 SQL 模拟器SQL Fiddle使用简介 有时候&#xff0c;我们想去验证 SQL语句&#xff0c;却缺少数据库环境&#xff0c;那该怎么办呢&#xff1f; 这时候在线 SQL 模拟器就有了用武之地。SQL 模拟器免安装&#xff0c;可以在网页直接运行 SQL 。 SQL Fiddle 支持 MySQL、Orac…

Stable Diffusion绘画系列【4】:可爱盲盒风人物

《博主简介》 小伙伴们好&#xff0c;我是阿旭。专注于人工智能、AIGC、python、计算机视觉相关分享研究。 ✌更多学习资源&#xff0c;可关注公-仲-hao:【阿旭算法与机器学习】&#xff0c;共同学习交流~ &#x1f44d;感谢小伙伴们点赞、关注&#xff01; 《------往期经典推…

数据结构——链式二叉树的实现(详解)

呀哈喽。我是结衣。 不知道大家的递归学到怎么样呢&#xff1f;如果大家的递归功底不是很好&#xff0c;那么我相信在学完这篇文章后大家一定会对递归有一个更深层次的了解的。 构造链式二叉树 在学习二叉树的基本操作前&#xff0c;需先要创建一棵二叉树&#xff0c;然后才能…

国标GB28181安防监控平台EasyCVR周界入侵AI算法检测方案

在城市管理和公共安全领域&#xff0c;安全视频监控的重要性日益凸显。AI视频智能分析平台基于深度学习和计算机视觉技术&#xff0c;利用AI入侵算法&#xff0c;能够实时、精准地监测周界入侵行为。 TSINGSEE青犀在视频监控及AI视频智能分析领域拥有深厚的技术积累和丰富的实…

2020年3月25日 Go生态洞察:Go、Go社区与疫情大流行

&#x1f337;&#x1f341; 博主猫头虎&#xff08;&#x1f405;&#x1f43e;&#xff09;带您 Go to New World✨&#x1f341; &#x1f984; 博客首页——&#x1f405;&#x1f43e;猫头虎的博客&#x1f390; &#x1f433; 《面试题大全专栏》 &#x1f995; 文章图文…

2024年最受欢迎的项目管理工具盘点

十大项目管理系统包括&#xff1a;1.产品研发项目管理工具&#xff1a;PingCode&#xff1b;2.通用项目协作工具&#xff1a;Worktile&#xff1b;3.开源项目管理系统&#xff1a;Redmine&#xff1b;4.IT/敏捷项目管理系统&#xff1a;Jira&#xff1b;5.免费个人项目管理&…

java基础进阶-线程池

1、线程池 线程池就是一个可以复用线程的技术。 2、应用场景 用户每发起一个请求&#xff0c;后台就需要创建一个新线程来处理&#xff0c;下次新任务来了肯定又要创建新线程处理的&#xff0c;而创建新线程的开销是很大的&#xff0c;并且请求过多时&#xff0c;肯定会产生大…

2023年【起重机械指挥】考试题库及起重机械指挥考试资料

题库来源&#xff1a;安全生产模拟考试一点通公众号小程序 2023年【起重机械指挥】考试题库及起重机械指挥考试资料&#xff0c;包含起重机械指挥考试题库答案和解析及起重机械指挥考试资料练习。安全生产模拟考试一点通结合国家起重机械指挥考试最新大纲及起重机械指挥考试真…

OpenSSL 使用AES对文件加解密

AES&#xff08;Advanced Encryption Standard&#xff09;是一种对称加密算法&#xff0c;它是目前广泛使用的加密算法之一。AES算法是由美国国家标准与技术研究院&#xff08;NIST&#xff09;于2001年发布的&#xff0c;它取代了原先的DES&#xff08;Data Encryption Stand…

设计师福利!2024在线图标设计网站推荐,不容错过的宝藏!

在当今竞争激烈的商业环境中&#xff0c;公司或个人品牌的视觉识别元素已经成为区分你和竞争对手的关键因素之一。一个独特而引人注目的标志可以深深扎根于人们的心中&#xff0c;并在消费者心中建立一个强烈的品牌印象。如果你正在寻找合适的工具来创建或改进你的标志&#xf…

Nginx系列-正向代理和反向代理

Nginx系列-正向代理和反向代理 文章目录 Nginx系列-正向代理和反向代理1. 三个对象2. 两种场景代理2.1. 正向代理2.2. 反向代理 3. 两种场景的对比3.1 为什么叫做反向代理3.2 正向代理和反向代理的作用 1. 三个对象 客户端&#xff1a;发出请求到代理&#xff0c;并接收代理的…

2019年11月20日 Go生态洞察:Go开发者调查启动

&#x1f337;&#x1f341; 博主猫头虎&#xff08;&#x1f405;&#x1f43e;&#xff09;带您 Go to New World✨&#x1f341; &#x1f984; 博客首页——&#x1f405;&#x1f43e;猫头虎的博客&#x1f390; &#x1f433; 《面试题大全专栏》 &#x1f995; 文章图文…

1128. 等价多米诺骨牌对的数量

力扣&#xff08;LeetCode&#xff09;官网 - 全球极客挚爱的技术成长平台备战技术面试&#xff1f;力扣提供海量技术面试资源&#xff0c;帮助你高效提升编程技能&#xff0c;轻松拿下世界 IT 名企 Dream Offer。https://leetcode.cn/problems/number-of-equivalent-domino-pa…

主机的具体权限规划:ACL的使用

目的&#xff1a;针对某一用户或某一组来设置特定权限需求&#xff0c;针对上&#xff0c;接着设置 ACL可以针对单一用户&#xff0c;文件&#xff0c;或者目录来进行rwx的权限设置&#xff0c;对于需要特殊权限的设置非常有帮助。 第一&#xff0c;查看文件系统是否支持&…