C++设计模式_01_设计模式简介(多态带来的便利;软件设计的目标:复用)
文章目录
- 本栏简介
- 1. 什么是设计模式
- 2. GOF 设计模式
- 3. 从面向对象谈起
- 4. 深入理解面向对象
- 5. 软件设计固有的复杂性
- 5.1 软件设计复杂性的根本原因
- 5.2 如何解决复杂性 ?
- 6. 结构化 VS. 面向对象
- 6.1 同一需求的分解写法
- 6.1.1 Shape1.h
- 6.1.2 MainForm1.cpp
- 6.2 同一需求的抽象的写法
- 6.2.1 shape2.h
- 6.2.2 MainForm2.cpp
- 6.3 两种方法的分析:
- 6.3.1 Shape*与Shape的使用
- 6.3.2 第一种设计和第二种设计对比
- 7. 软件设计的目标
本栏简介
本栏目将会介绍设计模式的相关内容,目标如下:
- 理解松耦合设计思想
- 掌握面向对象设计原则
- 掌握重构技法改善设计
- 掌握GOF 核心设计模式
1. 什么是设计模式
软件领域的设计模式是参考一个建筑学家提出的进行定义的。
“每一个模式描述了一个在我们周围不断重复
发生的问题以及该问题的解决方案
的核心。这样,你就能一次又一次地使用该方案而不必做重复劳动” -Christopher Alexander
也就是不用再重新开发轮子
2. GOF 设计模式
本栏目推荐的教材
如下,因为是四个人94年写的这本书,GOF也有四人团的说法。
-
历史性著作《设计模式:可复用面向对象软件的基础》一书中描述了
23种
经典面向对象设计模式,创立了模式在软件设计中的地位。
可复用
是设计模式的目标,面向对象
是具体的方法 -
由于《设计模式》一书确定了设计模式的地位,通常所说的设计模式隐含地表示
“面向对象设计模式”
。但这并不意味“设计模式"就等于“面向对象设计模式"。
3. 从面向对象谈起
面向对象的设计模式,很重要的一点就是从面向对象谈起。
面向对象背后藏着两种思维模型:
- 底层思维 (人和计算机之间的沟通,建立机器模型): 向下,如何把握机器底层从微观理解对象构造
- 语言构造
- 编译转换
- 内存模型
- 运行时机制 (异常处理、垃圾收集器进行内存管理)
底层思维可以帮助程序员建立机器模型,很多C++的高手也是从这一步开始的,但是光有底层思维还是不够,伴随着工作经验的增长,抽象思维会显得很重要。
- 抽象思维: 向上,如何将我们的周围世界抽象为程序代码
- 面向对象
- 组件封装
设计模式
- 架构模式
抽象思维可以帮助我们很好的关系代码的复杂度。
4. 深入理解面向对象
光有抽象思维而没有底层思维,很有可能代码是写不好的,所以这两种思维需要并重,而本栏目更倾向于抽象思维。
-
向下:深入理解三大面向对象机制
- 封装,隐藏内部实现
- 继承,复用现有代码
- 多态,改写对象行为
-
向上: 深刻把握面向对象机制所带来的抽象意义,理解如何使用这些机制来表达现实世界,掌握什么是“好的面向对象设计。
5. 软件设计固有的复杂性
首先需要谈抽象思维的背景,即软件设计固有的复杂性。
建筑商从来不会去想给一栋已建好的100层高的楼房底下再新修一个小地下室-这样做花费极大而且注定要失败。然而令人惊奇的是,软件系统的用户在要求作出类似改变时却不会仔细考虑,而且他们认为这只是需要简单编程的事。-Object-Oriented Analysis and Design with Applications
5.1 软件设计复杂性的根本原因
软件设计复杂性的根本原因即变化
。
- 客户需求的变化
- 技术平台的变化
- 开发团队的变化
- 市场环境的变化
…
5.2 如何解决复杂性 ?
- 分解:复用性差
- 人们面对复杂性有一个常见的做法:即
分而治之
,将大问题分解为多个小问题,将复杂问题分解为多个简单问题。
- 人们面对复杂性有一个常见的做法:即
- 抽象:统一处理,不用分而治之
- 更高层次来讲,人们处理复杂性有一个通用的技术,即抽象。由于不能掌握全部的复杂对象,我们选择忽视它的非本质细节而去处理泛化和理想化了的对象模型。
所有的设计模式都是围绕着抽象关键词进行变化。
下面通过代码来理解分解和抽象。
6. 结构化 VS. 面向对象
此处均为伪码:不要紧的删除,只保留能表达关键设计的代码
6.1 同一需求的分解写法
6.1.1 Shape1.h
class Point{
public:int x;int y;
};class Line{
public:Point start;Point end;Line(const Point& start, const Point& end){this->start = start;this->end = end;}};class Rect{
public:Point leftUp;int width;int height;Rect(const Point& leftUp, int width, int height){this->leftUp = leftUp;this->width = width;this->height = height;}};//增加
class Circle{};
上面的代码没有太遵循C++的编码规范。
例如:
public:int x;int y;
字段作为实现是细节,是需要定义为private的,但是那么写的话就会写的很长。
还有不同的Class需要放在独立的文件中,为了方便代码展示,就将不同的类放在了一个文件中。
6.1.2 MainForm1.cpp
假设有一个窗口MainForm
,以下程序是在界面上划线,画矩形等。
class MainForm : public Form {
private:Point p1; //描述鼠标移动留下的点的轨迹Point p2;vector<Line> lineVector;vector<Rect> rectVector;//改变vector<Circle> circleVector;public:MainForm(){//...}
protected:virtual void OnMouseDown(const MouseEventArgs& e);virtual void OnMouseUp(const MouseEventArgs& e);virtual void OnPaint(const PaintEventArgs& e); //界面刷新
};void MainForm::OnMouseDown(const MouseEventArgs& e){p1.x = e.X;p1.y = e.Y;//...Form::OnMouseDown(e);
}void MainForm::OnMouseUp(const MouseEventArgs& e){p2.x = e.X;p2.y = e.Y;if (rdoLine.Checked){Line line(p1, p2);lineVector.push_back(line);}else if (rdoRect.Checked){int width = abs(p2.x - p1.x);int height = abs(p2.y - p1.y);Rect rect(p1, width, height);rectVector.push_back(rect);}//改变else if (...){//...circleVector.push_back(circle);}//...this->Refresh();Form::OnMouseUp(e);
}//界面刷新的时候会被调用到,以下是针对不同图形的画法,主要关注业务逻辑
void MainForm::OnPaint(const PaintEventArgs& e){//针对直线for (int i = 0; i < lineVector.size(); i++){e.Graphics.DrawLine(Pens.Red,lineVector[i].start.x, lineVector[i].start.y,lineVector[i].end.x,lineVector[i].end.y);}//针对矩形for (int i = 0; i < rectVector.size(); i++){e.Graphics.DrawRectangle(Pens.Red,rectVector[i].leftUp,rectVector[i].width,rectVector[i].height);}//改变//针对圆形for (int i = 0; i < circleVector.size(); i++){e.Graphics.DrawCircle(Pens.Red,circleVector[i]);}//...Form::OnPaint(e);
}
以上即为分解的设计方法,
6.2 同一需求的抽象的写法
6.2.1 shape2.h
//
class Shape{
public:virtual void Draw(const Graphics& g)=0;virtual ~Shape() { } //虚析构函数作用:通过多态释放的时候,子类的析构函数才会被调用到
};class Point{
public:int x;int y;
};
//所有的继承推荐使用public,很少使用其他类型
class Line: public Shape{
public:Point start;Point end;Line(const Point& start, const Point& end){this->start = start;this->end = end;}//实现自己的Draw,负责画自己virtual void Draw(const Graphics& g){g.DrawLine(Pens.Red, start.x, start.y,end.x, end.y);}};class Rect: public Shape{
public:Point leftUp;int width;int height;Rect(const Point& leftUp, int width, int height){this->leftUp = leftUp;this->width = width;this->height = height;}//实现自己的Draw,负责画自己virtual void Draw(const Graphics& g){g.DrawRectangle(Pens.Red,leftUp,width,height);}};//增加
class Circle : public Shape{
public://实现自己的Draw,负责画自己virtual void Draw(const Graphics& g){g.DrawCircle(Pens.Red,...);}};
6.2.2 MainForm2.cpp
class MainForm : public Form {
private:Point p1;Point p2;//针对所有形状,没必要像上面设计数据结构vector<Shape*> shapeVector; public:MainForm(){//...}
protected:virtual void OnMouseDown(const MouseEventArgs& e);virtual void OnMouseUp(const MouseEventArgs& e);virtual void OnPaint(const PaintEventArgs& e);
};void MainForm::OnMouseDown(const MouseEventArgs& e){p1.x = e.X;p1.y = e.Y;//...Form::OnMouseDown(e);
}void MainForm::OnMouseUp(const MouseEventArgs& e){p2.x = e.X;p2.y = e.Y;if (rdoLine.Checked){//需要塞入堆对象,需要在最终进行释放,为了表达的方便性,对有些内存管理的就不做展示shapeVector.push_back(new Line(p1,p2));}else if (rdoRect.Checked){int width = abs(p2.x - p1.x);int height = abs(p2.y - p1.y);shapeVector.push_back(new Rect(p1, width, height));}//改变else if (...){//...shapeVector.push_back(circle);}//...this->Refresh();Form::OnMouseUp(e);
}void MainForm::OnPaint(const PaintEventArgs& e){//针对所有形状for (int i = 0; i < shapeVector.size(); i++){shapeVector[i]->Draw(e.Graphics); //多态调用,各负其责:根据存储的实际类型进行调用,接口一样,但实现不一样}//...Form::OnPaint(e);
}
6.3 两种方法的分析:
6.3.1 Shape*与Shape的使用
MainForm2.cpp中:需要多态,此处虽然是Shape*
类型,但是可能真正塞得是Line
、Rect
、Circle
类。
如果不使用Shape*
,而使用Shape
会导致对象切割,假如传的是Line,就会将其切割为小对象,后期会再进行梳理,此处只要理解需要使用shape指针来表达多态性,而不能使用shape对象。
//针对所有形状,没必要像上面设计数据结构vector<Shape*> shapeVector;
MainForm1.cpp中:存储的是对象,不需要多态
vector<Line> lineVector;vector<Rect> rectVector;
6.3.2 第一种设计和第二种设计对比
为了证实哪种设计更好,我们需要假设在客户需求发生变化时,对程序员来说哪种方式复用性更强。
假设存在一种变化,比如客户需求变化,客户需要增加实现圆,circle
,上面的代码已经是增加了circle的,可以对两种方法进行对比,可以发现在第一种设计方法的改变内容要大于第二种方法,也就是第二种设计方式重用性得到了提升。
7. 软件设计的目标
什么是好的软件设计 ?软件设计的金科玉律:
复用性!
大家必须深刻理解 抽象
这种设计模式和目标,后期介绍利用抽象的设计思想,针对不同领域的不同问题,提出不同的模式进行解决。
相关文章:

C++设计模式_01_设计模式简介(多态带来的便利;软件设计的目标:复用)
文章目录 本栏简介1. 什么是设计模式2. GOF 设计模式3. 从面向对象谈起4. 深入理解面向对象5. 软件设计固有的复杂性5.1 软件设计复杂性的根本原因5.2 如何解决复杂性 ? 6. 结构化 VS. 面向对象6.1 同一需求的分解写法6.1.1 Shape1.h6.1.2 MainForm1.cpp 6.2 同一需求的抽象的…...
Docker技术--WordPress博客系统部署初体验
如果使用的是传统的项目部署方式,你要部署WordPress博客系统,那么你需要装备一下的环境,才可以部署使用。 -1:操作系统linux -2:PHP5.6或者是更高版本环境 -3:MySQL数据环境 -4:Apache环境 但是如果使用Docker技术,那么就只需要进行如下的几行简单的指令: docker run …...
提高代码可读性和可维护性的命名建议
当进行接口自动化测试时,良好的命名可以提高代码的可读性和可维护性。以下是一些常用的命名建议: 变量和函数命名: 使用具有描述性的名称,清晰地表达变量或函数的用途和含义。使用小写字母和下划线来分隔单词,例如 log…...

Docker基础入门:Docker网络与微服务项目发布
Docker基础入门:Docker网络与微服务项目发布 一、前言二、Docker0理解2.1 ip a查看当前网络环境2.2 实战--启动一个tomact01容器(查看网络环境)2.3 实战--启动一个tomact02容器(查看网络环境)2.4 容器与容器之间的通信…...

Docker安装详细步骤
Docker安装详细步骤 1、安装环境准备 主机:192.168.40.5 zch01 设置主机名 # hostnamectl set-hostname zch01 && bash 配置hosts文件 [root ~]# vi /etc/hosts 添加如下内容: 192.168.40.5 zch01 关闭防火墙 [rootzch01 ~]# systemct…...

十六、pikachu之SSRF
文章目录 1、SSRF概述2、SSRF(URL)3、SSRF(file_get_content) 1、SSRF概述 SSRF(Server-Side Request Forgery:服务器端请求伪造):其形成的原因大都是由于服务端提供了从其他服务器应用获取数据的功能&…...

最新PHP短网址生成系统/短链接生成系统/URL缩短器系统源码
全新PHP短网址系统URL缩短器平台,它使您可以轻松地缩短链接,根据受众群体的位置或平台来定位受众,并为缩短的链接提供分析见解。 系统使用了Laravel框架编写,前后台双语言使用,可以设置多域名,还可以开设套…...

漱玉平民大药房:多元化药店变革的前夜
作者 | 王聪彬 编辑 | 舞春秋 来源 | 至顶网 本文介绍了漱玉平民大药房在药品零售领域的数字化转型和发展历程。通过技术创新, 漱玉平民 建设了覆盖医药全生命周期的大健康生态圈,采用混合云架构和国产分布式数据库 TiDB,应对庞大的会员数据处…...

如何实现AI的矢量数据库
推荐:使用 NSDT场景编辑器 助你快速搭建3D应用场景 然而,人工智能模型有点像美食厨师。他们可以创造奇迹,但他们需要优质的成分。人工智能模型在大多数输入上都做得很好,但如果它们以最优化的格式接收输入,它们就会真正…...
Java与Modbus-TCP/IP网络通讯
1.需求样例 举例5:浮点数参数读取(读取温度测量值)查看参数列表,温度测量值地址为320,根据Modbus协议,读取参数地址转换为16进制为:00H A0H,读取长度为2个字:00H 02H。 …...
音视频 ffmpeg命令图片与视频互转
截取一张图片 ffmpeg -i test.mp4 -y -f image2 -ss 00:00:02 -vframes 1 -s 640x360 test.jpg ffmpeg -i test.mp4 -y -f image2 -ss 00:00:02 -vframes 1 -s 640x360 test.bmp -i 输入 -y 覆盖 -f 格式 image2 一种格式 -ss 起始值 -vframes 帧 如果大于1 那么 输出加%03d t…...

C++的基类和派生类构造函数
基类的成员函数可以被继承,可以通过派生类的对象访问,但这仅仅指的是普通的成员函数,类的构造函数不能被继承。构造函数不能被继承是有道理的,因为即使继承了,它的名字和派生类的名字也不一样,不能成为派生…...
C语言中对json格式数据的解析和封装
首先需要调库:#include <cJSON.h> Json的数据结构介绍: /* The cJSON structure: */ typedef struct cJSON { /*next/prev允许您遍历数组/对象链。或者,使用GetArraySize/GetArrayItem/GetObjectItem */ struct cJSON *next; struct c…...

RT-Thread自动初始化机制
自动初始化机制是指初始化函数不需要被显示调用,只需要在函数定义处通过宏定义的方式进行申明,就会在系统启动过程中被执行。 int rt_hw_usart_init(void) {rt_hw_serial_register(&serial1, "uart1",RT_DEVICE_FLAG_RDWR | RT_DEVICE_FL…...

在本地搭建Jellyfin影音服务器,支持公网远程访问影音库的方法分享
文章目录 1. 前言2. Jellyfin服务网站搭建2.1. Jellyfin下载和安装2.2. Jellyfin网页测试 3.本地网页发布3.1 cpolar的安装和注册3.2 Cpolar云端设置3.3 Cpolar本地设置 4.公网访问测试5. 结语 1. 前言 随着移动智能设备的普及,各种各样的使用需求也被开发出来&…...
强盛集团面试题实战(持续更新)
目录 第一章、面试题1.1)王:8月301.2)1.3) 第二章、2.1)2.2)2.3) 第三章、3.1)3.2)3.3) 第四章、4.1)4.2)4.3) 友情提醒:先…...

golang 协程的实现原理
核心概念 要理解协程的实现, 首先需要了解go中的三个非常重要的概念, 它们分别是G, M和P, 没有看过golang源代码的可能会对它们感到陌生, 这三项是协程最主要的组成部分, 它们在golang的源代码中无处不在. G (goroutine) G是goroutine的头文字, goroutine可以解释为受管理的…...

go gin 参数绑定常用验证器
https://pkg.go.dev/github.com/go-playground/validator/v10#readme-baked-in-validations min 最小max 最大len 长度限制gt 大于eq 等于ne 不等于eqfield 与某个字段值一样nefield 与某个字段值不一样oneof 枚举 ,以空格分开startswithendswithdive 数组 package mainimpor…...

多用户商城系统常见的安全性和数据保护措施有哪些?
电子商务的迅速发展,越来越多的企业选择搭建多用户商城系统来扩展业务。然而,随之而来的是对数据安全和保护的日益关注。在选择多用户商城系统时,我们需要考虑一系列的安全性和数据保护措施,以确保商城系统的稳定性和用户数据的完…...

如何在WSL上导入任何Linux发行版
文章目录 一、准备1. 开启WSL相关功能2. 升级WSL3. 设置默认的wsl版本 二、通过 Microsoft Store 安装 Linux1. 查看 Microsoft Store 有哪些可安装的 Linux 发行版2. 安装 Linux 发行版3. 查看已安装的 Linux 发行版4. 启动Linux发行版 三、通过Linux发行商提供的tar文件安装1…...
在HarmonyOS ArkTS ArkUI-X 5.0及以上版本中,手势开发全攻略:
在 HarmonyOS 应用开发中,手势交互是连接用户与设备的核心纽带。ArkTS 框架提供了丰富的手势处理能力,既支持点击、长按、拖拽等基础单一手势的精细控制,也能通过多种绑定策略解决父子组件的手势竞争问题。本文将结合官方开发文档,…...

DAY 47
三、通道注意力 3.1 通道注意力的定义 # 新增:通道注意力模块(SE模块) class ChannelAttention(nn.Module):"""通道注意力模块(Squeeze-and-Excitation)"""def __init__(self, in_channels, reduction_rat…...
TRS收益互换:跨境资本流动的金融创新工具与系统化解决方案
一、TRS收益互换的本质与业务逻辑 (一)概念解析 TRS(Total Return Swap)收益互换是一种金融衍生工具,指交易双方约定在未来一定期限内,基于特定资产或指数的表现进行现金流交换的协议。其核心特征包括&am…...
linux 下常用变更-8
1、删除普通用户 查询用户初始UID和GIDls -l /home/ ###家目录中查看UID cat /etc/group ###此文件查看GID删除用户1.编辑文件 /etc/passwd 找到对应的行,YW343:x:0:0::/home/YW343:/bin/bash 2.将标红的位置修改为用户对应初始UID和GID: YW3…...
Axios请求超时重发机制
Axios 超时重新请求实现方案 在 Axios 中实现超时重新请求可以通过以下几种方式: 1. 使用拦截器实现自动重试 import axios from axios;// 创建axios实例 const instance axios.create();// 设置超时时间 instance.defaults.timeout 5000;// 最大重试次数 cons…...

前端开发面试题总结-JavaScript篇(一)
文章目录 JavaScript高频问答一、作用域与闭包1.什么是闭包(Closure)?闭包有什么应用场景和潜在问题?2.解释 JavaScript 的作用域链(Scope Chain) 二、原型与继承3.原型链是什么?如何实现继承&a…...
AGain DB和倍数增益的关系
我在设置一款索尼CMOS芯片时,Again增益0db变化为6DB,画面的变化只有2倍DN的增益,比如10变为20。 这与dB和线性增益的关系以及传感器处理流程有关。以下是具体原因分析: 1. dB与线性增益的换算关系 6dB对应的理论线性增益应为&…...

深入浅出深度学习基础:从感知机到全连接神经网络的核心原理与应用
文章目录 前言一、感知机 (Perceptron)1.1 基础介绍1.1.1 感知机是什么?1.1.2 感知机的工作原理 1.2 感知机的简单应用:基本逻辑门1.2.1 逻辑与 (Logic AND)1.2.2 逻辑或 (Logic OR)1.2.3 逻辑与非 (Logic NAND) 1.3 感知机的实现1.3.1 简单实现 (基于阈…...
解决:Android studio 编译后报错\app\src\main\cpp\CMakeLists.txt‘ to exist
现象: android studio报错: [CXX1409] D:\GitLab\xxxxx\app.cxx\Debug\3f3w4y1i\arm64-v8a\android_gradle_build.json : expected buildFiles file ‘D:\GitLab\xxxxx\app\src\main\cpp\CMakeLists.txt’ to exist 解决: 不要动CMakeLists.…...

数据结构:递归的种类(Types of Recursion)
目录 尾递归(Tail Recursion) 什么是 Loop(循环)? 复杂度分析 头递归(Head Recursion) 树形递归(Tree Recursion) 线性递归(Linear Recursion)…...