C++ :设计模式实现

文章目录

      • 原则
        • 单一职责原则
        • 开闭原则
        • 依赖倒置原则
        • 接口隔离原则
        • 里氏替换原则
      • 设计模式
        • 单例模式
        • 观察者模式
        • 策略模式
        • 代理模式

原则

单一职责原则

定义:
即一个类只负责一项职责

问题:
类 T 负责两个不同的职责:职责 P1,职责 P2。当由于职责 P1 需求发生改变而需
要修改类 T 时,有可能会导致原本运行正常
的职责 P2 功能发生故

解决:
将类 T 分成两个不同的类来实现。

开闭原则

定义:
如类、模块和函数应该对扩展开放,对修改关闭

问题:
在软件的生命周期内,因为变化、升级和维护等原因需要对软件原有代码进行修改
时,可能会给旧代码中引入错误,也可能会使我们不得不对整个功能进行重构,并且需
要原有代码经过重新测试

解决:
当软件需要变化时,尽量通过扩展软件实体的行为来实现变化

依赖倒置原则

定义:
高层模块不应该依赖低层模块,二者都应该依赖其抽象;抽象不应该依赖细节;细节应该依赖抽象

问题:
类 A 直接依赖类 B,假如要将类 A 改为依赖类 C,则必须通过修改类 A 的代码来达成。这种场景下,类 A 一般是高层模块,负责复杂的业务逻辑;类 B 和类 C 是低层模
块,负责基本的原子操作;假如修改类 A,会给程序带来不必要的风

解决:
将类 A 修改为依赖接口 I,类 B 和类 C 各自实现接口 I,类 A 通过接口 I 间接与类 B或者类 C 发生联系,则会大大降低修改类 A 的几率
在这里插入图片描述

接口隔离原则

定义:
客户端不应该依赖它不需要的接口;一个类对另一个类的依赖应该建立在最小的接口上

问题:
类 A 通过接口 I 依赖类 B,类 C 通过接口 I 依赖类 D,如果接口 I 对于类 A 和类 B来说不是最小接口,则类 B 和类 D 必须去实现他们不需要的方法

解决:
将臃肿的接口 I 拆分为独立的几个接口,类 A 和类 C 分别与他们需要的接口建立依
赖关系。也就是采用接口隔离原则
在这里插入图片描述

里氏替换原则

定义:
所有引用基类的地方必须能透明地使用其子类的对象,子类可以扩展父类的功能,但不能改变父类原有的功能。

问题:
当使用继承时,遵循里氏替换原则。类 B 继承类 A 时,除添加新的方法完成新增
功能 P2 外,尽量不要 shadow(遮盖)父类 A 的方法。

解决:
子类中可以增加自己特有的方法;
子类可以实现父类的抽象方法,但不能覆盖父类的非抽象方法(多态实现)

设计模式

单例模式

定义:
确保某一个类只有一个实例, 而且自行实例化并向整个系统提供这个实例
问题:
几个不同的客户对象需要引用同样的对象,你希望确保自己拥有的这种对象不超过一个
解决:

  1. 在 Singleton 中添加静态成员,初始化为NULL
  2. 添加静态成员方法,若静态成员变量为 NULL,则实始化并返回
  3. 将构造成员,设为 private 或是 protected,这样只能通过静态方法实例化

类关系图:

在这里插入图片描述
实现:

#include <iostream>
using namespace std;
class Singleton
{
public:static Singleton* getInstance(){if(_ins == NULL)_ins = new Singleton;return _ins;}static void releaseInstance(){if(_ins != NULL){delete _ins;_ins = NULL;}}void run(){cout<<"test singleton"<<endl;}
private:Singleton(){}~Singleton(){}Singleton(const Singleton &){}Singleton & operator=(const Singleton&){}static Singleton * _ins;
};
Singleton * Singleton::_ins = NULL;
int main()
{Singleton * ps = Singleton::getInstance();ps->run();Singleton::releaseInstance();return 0;
}
观察者模式

定义:
定义对象间一种一对多的依赖关系, 使得每当一个对象改变状态, 则所有依赖于它的对象都会得到通知并被自动更新,也成为发布-订阅模式
问题:
当某个事件发生变化时,你需要向一系列对象发出通知,而这个对象的列表也是不断变化的。
解决:
Observer 将监视某个事件上的责任委托给一个中心对象:Subject ,当事件发生时,Subject 告诉 Observer“你关心的事件发生了”

类关系图:
在这里插入图片描述

实现:

// 观察者模式实现 时间更新

#include <iostream>
#include <list>
#include <algorithm>
using namespace std;// 观察者模式,又称为订阅模式 ,比如关注的公众号更新会自动发送到手机上class Observer
{
public:virtual void update(int hour, int min ,int sec) = 0;
};class Subject
{
public:virtual void registerObserver(Observer * ob) = 0;virtual void removeObserver(Observer *ob) = 0;virtual void notify() = 0;
protected:list<Observer*> observerList;
};class PekingTimeSubject:public Subject
{
public:void setTimer(int hour, int min, int sec){_hour = hour;_min = min;_sec = sec;notify();}void registerObserver(Observer * ob){observerList.push_back(ob);}void removeObserver(Observer *ob){//observerList.erase(find(observerList.begin(),observerList.end(),ob));observerList.remove(ob);}void notify() // 此函数必须在子类中实现,因为,要接触数据。{list<Observer*>::iterator itr = observerList.begin();for(; itr != observerList.end(); itr++){(*itr)->update(_hour,_min,_sec);}}
private:int _hour;int _min;int _sec;
};
class AmericaTimerObserver:public Observer
{
public:void update(int hour, int min ,int sec){_hour = hour;_min = min;_sec = sec;dis();}void dis(){cout<<"America Time is update"<<endl;cout<<"H:"<<_hour<<" M:"<<_min<<" S:"<<_sec<<endl;}
private:int _hour;int _min;int _sec;
};
class JapanTimerObserver:public Observer
{
public:void update(int hour, int min ,int sec){_hour = hour;_min = min;_sec = sec;dis();}void dis(){cout<<"Japan Time is update"<<endl;cout<<"H:"<<_hour<<" M:"<<_min<<" S:"<<_sec<<endl;}
private:int _hour;int _min;int _sec;
};
int main()
{PekingTimeSubject *bj = new PekingTimeSubject;JapanTimerObserver *jp = new JapanTimerObserver;AmericaTimerObserver *am = new AmericaTimerObserver;bj->registerObserver(jp);bj->registerObserver(am);bj->setTimer(10,20,30);bj->removeObserver(jp);bj->setTimer(1,2,3);return 0;
}
策略模式

定义:
提供便利的的功能/算法切换

问题:
将变动的策略写死在,固定的代码中。不利于更改和维护

解决:
将策略抽象出来,进行封装
Strategy(策略): 所有支持的算法的公共接口
Context(上下文)

类关系图:
在这里插入图片描述

实现:
// 实现游戏人物通过相同按键切换武器

#include <iostream>using namespace std;// 定义了一组算法,进行封装class Weapon
{
public:virtual void use() = 0;
};class Knife :public Weapon
{
public:void use(){cout<<" dao ni "<<endl;}
};
class Gun :public Weapon
{
public:void use(){cout<<" tutu ni "<<endl;}
};class CCsprite
{
public:CCsprite(Weapon *k){_w = k;}void changWeapon(Weapon *w){_w = w;}void faighting(){_w->use();}
protected:Weapon *_w;
};
int main()
{Knife k;CCsprite character(&k);character.faighting();Gun g;character.changWeapon(&g);character.faighting();character.changWeapon(&k);character.faighting();return 0;
}
代理模式

定义:
(为其他对象
提供一种代理以控制对这个对象的访问
问题:
对一个对象进行访问控制的一个原因是为了只有在我们确实需要这个对象时才对它进行创建和初始化
解决:
对被访问的提供一个壳
Subject 抽象主题角色
ConcreteSubject 具体主题角色也叫做被委托角色、 被代理角色,是业务逻辑的具体执行者
ProxySubject 代理主题角色,负责对真实角色的应用

类关系图:
在这里插入图片描述

实现:
//模拟加载页面中先显示文字,后显示图片的过程

#include <iostream>
#include <unistd.h>
using namespace std;// LargeImage 为自己,显示图片,需要先构造6s
// ProxyLargeImage 为它的代理,他们都继承虚基类ImageSubject,代理LargeImage调用图像
class Text
{
public:void showText(){cout<<"wed text is show "<<endl;}
};
class ImageSubject
{
public:virtual void showPicture() = 0;
};class LargeImage:public ImageSubject
{
public:LargeImage(){sleep(6);}virtual void showPicture(){cout<<"xiang ri kui \n"<<endl;}
};class ProxyLargeImage:public ImageSubject
{
public:ProxyLargeImage():li(NULL){}void showPicture(){if(li == NULL)li = new LargeImage;li->showPicture();}
protected:LargeImage *li;
};
class Document
{
public:Document(){_t = new Text; //1_i = new ProxyLargeImage; //1000}void print(){_t->showText();_i->showPicture();}Text *_t;ProxyLargeImage *_i;
};
int main(int argc, char *argv[])
{Document doc;doc.print(); // 一分部加载文件字,一部分加载图片return 0;
}

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

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

相关文章

Tomcat源码解析——一次请求的处理流程

在上一篇文章中&#xff0c;我们知道Tomcat在启动后&#xff0c;会在Connector中开启一个Acceptor(接收器)绑定线程然后用于监听socket的连接&#xff0c;那么当我们发出请求时&#xff0c;第一步也就是建立TCP连接&#xff0c;则会从Acceptor的run方法处进入。 Acceptor&…

使用CSS+HTML完成导航栏

HTML <!DOCTYPE html> <html lang"en"> <head> <meta charset"UTF-8"> <meta name"viewport" content"widthdevice-width, initial-scale1.0"> <title>导航栏示例</title> &l…

07 内核开发-避免命名冲突经验技巧分享

07 内核开发-避免命名冲突经验技巧分享 目录 07 内核开发-避免命名冲突经验技巧分享 1.如何在内核开发过程中&#xff0c;避免命名冲突 2. 背景 3.避免方法 4.了解下 文件/proc/kallsyms 5.总结 课程简介&#xff1a; Linux内核开发入门是一门旨在帮助学习者从最基本的…

java-异常

一、异常的概念及分类 Exception&#xff1a;异常&#xff0c;代表程序可能出现的问题 Exception分为两类&#xff1a; 1、运行时异常&#xff1a;RuntimeException以及其子类&#xff0c;编译阶段不会出现异常提醒&#xff0c;在运行阶段会出现异常提醒 2、编译时异常&…

基于SpringBoot+Vue网上商城系统的设计与实现

系统介绍 随着社会的不断进步与发展&#xff0c;人们经济水平也不断的提高&#xff0c;于是对各行各业需求也越来越高。特别是从2019年新型冠状病毒爆发以来&#xff0c;利用计算机网络来处理各行业事务这一概念更深入人心&#xff0c;由于用户工作繁忙的原因&#xff0c;去商…

抽象工厂模式设计实验

【实验内容】 楚锋软件公司欲开发一套界面皮肤库&#xff0c;可以对 Java 桌面软件进行界面美化。为了保护版权&#xff0c;该皮肤库源代码不打算公开&#xff0c;而只向用户提供已打包为 jar 文件的 class 字节码文件。用户在使用时可以通过菜单来选择皮肤&#xff0c;不同的…

Java数据类型以及范围

数据类型&#xff1a; 取值范围&#xff1a; 取值&#xff1a;

磁性呼吸传感技术与机器学习结合在COVID-19审断中的应用

介绍 呼吸不仅是人类生存的基础&#xff0c;而且其模式也是评估个体健康状态的关键指标。异常的呼吸模式往往是呼吸系统疾病的一个警示信号&#xff0c;包括但不限于慢性阻塞性肺病&#xff08;COPD&#xff09;、阻塞性睡眠呼吸暂停&#xff08;OSA&#xff09;、肺炎、囊性纤…

idea连接Docker数据库

我们在docker下创建了数据库&#xff0c;想要更方便的查看和操作该数据库&#xff0c;idea和DataGrip或者其他软件都可以。在数据库连接时需要填写数据库名字&#xff0c;主机&#xff0c;端口&#xff0c;数据库用户名和密码。 输入之后先不要点击OK和按Enter键&#xff0c;我…

GAN详解,公式推导解读,详细到每一步的理论推导

在看这一篇文章之前&#xff0c;希望熟悉掌握熵的知识&#xff0c;可看我写的跟熵相关的一篇博客https://blog.csdn.net/m0_59156726/article/details/138128622 1. GAN 原始论文&#xff1a;https://arxiv.org/pdf/1406.2661.pdf 放一张GAN的结构&#xff0c;如下&#xff1…

Linux:动静态库介绍

动静态库 库的介绍开发环境 & 编译器库存在的意义库的实现库的命名静态库制作和使用总结 动态库的制作和使用动态库的使用方法方法一方法二方法三 库加载问题静态库加载问题动态库的加载问题与位置无关码 C/C静态库下载方式 库的介绍 静态库&#xff1a;程序在编译链接的时…

计算机网络---第十一天

生成树协议 stp作用&#xff1a; 作用&#xff1a;stp用于解决二层环路问题。 BPDU&#xff1a; 含义&#xff1a;桥协议数据单元&#xff0c;用于传递stp协议相关报文 分类&#xff1a;配置bpdu---用于传递stp的配置信息 tcn bpdu---用于通告拓扑变更信息 包含信息&…

数据库主键ID自增,两种方法获取插入数据库那条数据自动生成的主键ID值

目录 1. 前言 2. 适用于 MyBatis 框架 2.1 获取单条插入语句生成的ID 2.2 获取集合插入生成的多条数据的ID 3. 适用于 MyBatisPlus 框架 3.1 获取单条数据插入生成的ID 3.2 获取集合插入数据生成的多条数据的ID 4. 小结 1. 前言 在开发过程中&#xff0c;我们可能会遇…

OpenCompass 大模型评测实战——作业

OpenCompass 大模型评测实战——作业 一、基础作业1.1、使用 OpenCompass 评测 internlm2-chat-1_8b 模型在 C-Eval 数据集上的性能1.1.1、安装基本环境1.1.2、解压数据集1.1.3、查看支持的数据集和模型1.1.4、启动评测 二、进阶作业2.1、将自定义数据集提交至OpenCompass官网 …

2024春季春日主题活动策划方案

2024解冻派对“春日浪漫”主题活动策划方案-32P 方案页码&#xff1a;32页 文件格式&#xff1a;pptx 方案简介&#xff1a; 春来一季&#xff0c;新生欢喜 花香丨微风丨阳光 活动唤起【春日浪漫记忆】&#xff01; 年轻人不一样的派对活动 可以与朋友/小朋友/家人互动…

深度学习-线性代数

目录 标量向量矩阵特殊矩阵特征向量和特征值 标量由只有一个元素的张量表示将向量视为标量值组成的列表通过张量的索引来访问任一元素访问张量的长度只有一个轴的张量&#xff0c;形状只有一个元素通过指定两个分量m和n来创建一个形状为mn的矩阵矩阵的转置对称矩阵的转置逻辑运…

03-JAVA设计模式-访问者模式

访问者模式 什么是访问者模式 访问者模式&#xff08;Visitor Pattern&#xff09;是软件设计模式中的一种行为模式&#xff0c;它用于将数据结构中的元素与操作这些元素的操作解耦。这种模式使得可以在不修改数据结构的情况下添加新的操作。 在访问者模式中&#xff0c;我们…

图文教程 | Git安装配置、常用命令大全以及常见问题

前言 因为多了一台电脑&#xff0c;平时写一些代码&#xff0c;改一些文件&#xff0c;用U盘存着转来转去特别麻烦。于是打算用Git管理我的文件&#xff0c;方便在两个终端之间传输数据啥的。也正好给新电脑装好Git。 &#x1f4e2;博客主页&#xff1a;程序源⠀-CSDN博客 &…

HFSS端口介绍2---波端口

前面我们讨论了Lumped Port设定相关的内容,这节我们继续讨论Wave Port(波端口)使用相关的问题。 波端口使用范围 封闭结构:如波导、同轴电缆等 包含多个传播模式的模型 端口平面在求解区域外的模型 模型中包含均匀的波导或者传输线结构 波端口的大小 对于封闭的传输线结构:边…

视频教程下载:用ChatGPT的 API 开发AI应用指南

通过这门关于 OpenAI API 和 ChatGPT API 的全面课程&#xff0c;在您的应用中释放人工智能的力量。随着人工智能技术的快速发展&#xff0c;比以往任何时候都更重要的是保持领先地位&#xff0c;并为您的项目利用这些尖端工具。在本课程中&#xff0c;您将深入了解人工智能驱动…