C++设计模式_02_面向对象设计原则

news2024/12/25 10:11:56

文章目录

  • 1. 面向对象设计,为什么?
  • 2. 重新认识面向对象
  • 3. 面向对象设计原则
    • 3.1 依赖倒置原则(DIP)
    • 3.2 开放封闭原则(OCP )
    • 3.3 单一职责原则( SRP )
    • 3.4 Liskov 替换原则 ( LSP )
    • 3.5 接口隔离原则 ( ISP )
    • 3.6 优先使用对象组合,而不是类继承
    • 3.7 封装变化点
    • 3.8 针对接口编程,而不是针对实现编程
  • 4. 将设计原则提升为设计经验

1. 面向对象设计,为什么?

变化是复用的天敌!面向对象设计或者说使用了抽象原则的面向对象设计最大的优势在于:
抵御变化!

2. 重新认识面向对象

之前我们对面向对象的理解,包括封装、继承、多态,这是属于面向对象的底层思维理解。从抽象思维进行如下理解:

  • 理解隔离变化
    • 从宏观层面来看,面向对象的构建方式更能适应软件的变化,能将变化所带来的影响减为最小
  • 各司其职
    • 从微观层面来看,面向对象的方式更强调各个类的“责任”
    • 由于需求变化导致的新增类型不应该影响原来类型的实现-所谓各负其责(多态更多的使用)
  • 对象是什么?
    • 从语言实现层面来看,对象封装了代码和数据。
    • 从规格层面讲,对象是一系列可被使用的公共接口。
    • 从概念层面讲,对象是某种拥有责任的抽象

3. 面向对象设计原则

设计原则要比具体某一个模式更为重要,后期讲到的模式当下已经不流行,某些已经被有些语言模式替代,甚至忘了也不重要。有了这些设计原则可以理解其他的模式,甚至可以设计自己的模式。

有一种错误的观点就是将设计模式当做算法去学习,后期可以看到两段代码非常相似,但是表达的模式是完全不同的,也有一种可能就是代码千差万别,但是设计模式是一样的。

所有的设计模式都是依赖这里所讲的设计原则,如果违背了设计原则,这种设计模式就是错误的。设计原则是一把尺子,可以用来衡量软件设计品质。
学习设计模式,一定不要只看代码的调用流程,如果你了解虚函数的调用机制,23种设计模式可以很快看懂,但是要掌握设计模式,就需要从设计原则的方向出发。

3.1 依赖倒置原则(DIP)

  • 高层模块(稳定)不应该依赖于低层模块(变化),二者都应该依赖于抽象(稳定) 。
  • 抽象(稳定)不应该依赖于实现细节(变化) ,实现细节应该依赖于抽象(稳定)。

上面的话怎么理解呢,上篇中Mainform,依赖于Line和Rec。示意图如下图所示:箭头指向的代表了依赖
在这里插入图片描述
Mainform属于高层模块,Line和Rec属于底层模块,由于底层模块是容易变化的,高层模块是相当文档的,稳定的东西依赖于变化的东西,稳定的也就变成不稳定的,而应该将二者都依赖于抽象。

转换为第二种方式,示意图如下图所示:
在这里插入图片描述
Mainform依赖于Shape,Line和Rec也是依赖于Shape,Shape也就是上面讲到的抽象。

有些人喜欢在抽象类中使用子类,那就违背了抽象(稳定)不应该依赖于实现细节(变化) ,这是因为抽象应该是稳定的,如果依赖于一个实现细节,这个实现细节是容易变化的,就会变成,稳定的依赖于变化。

这里最关键的地方就是Line和Rec不稳定,容易变化,导致Mainform不稳定。

依赖倒置原则( DIP)原则贯穿于所有的设计模式,这种原则是十分依赖抽象类的。

3.2 开放封闭原则(OCP )

  • 对扩展开放,对更改封闭。
  • 类模块应该是可扩展的,但是不可修改。

首先举一个现实生活的例子,有一个需求,需要做一套桌椅板凳,将需求提交给一个木器加工厂,加工过程中,使用场景是在一个办公大楼,所有的办公家具需要达到一个几级阻燃的标准,现在将新的需求告诉加工厂,木器加工厂如何应对呢?有两种应对方法:

  1. 将做好的东西扔掉,重新再弄,这是一种改变的做法,不太好,浪费程度很大。
  2. 聪明的木器加工厂,只需要刷一层防火涂料就可以了,他已经做了一些工作,只需要在前面基础上刷一层防火涂料,这是一种扩展的做法。

从代码的角度分析,上篇的代码中就使用了开放封闭原则,第二种方法就是以扩展的方式来应对需求的变化。

当需求变化的情况下,不要想着去更改,应该想着增加一些东西来应对需求的变化。

3.3 单一职责原则( SRP )

  • 一个类应该仅有一个引起它变化的原因。
  • 变化的方向隐含着类的责任

有的时候,你会发现一个类会写的很臃肿,当一个类中放了太多的功能,也就隐含了多个责任,当隐含多个责任的时候就会出毛病,多个责任将你往不同的方向拉扯,用子类的时候,一会这个方向,一会那个方向。后期会讲到桥模式装饰模式都会遇到类的责任问题,类的责任问题通常情况下,表现不是那么强烈,当遇到特殊的场合,类的责任是非常重要。

3.4 Liskov 替换原则 ( LSP )

  • 子类必须能够替换它们的基类(IS-A)
  • 继承表达类型抽象。

作为子类,所有需要父类的地方,都可以将子类传过去进行使用,这是天经地义的,但是具体实现时,就是会有人打破这个原则。
例如,有些程序员朋友喜欢在继承父类的子类中,发现父类中的三个方法,子类无法使用的,这个时候会在子类的3个方法中throw exception,这个做法就不对了。这就说明子类没有继承父类,或者说有可能是组合关系。

3.5 接口隔离原则 ( ISP )

  • 不应该强迫客户程序依赖它们不用的方法。
  • 接口应该小而完备。

“接口应该小而完备”中的意思就是,不要把不必要的方法,public出去。如果只是子类使用的话,就protected。如果本类使用的话就private。如果真正有必要暴露的情况下,才将其public出去。
当你无节制的public出去,就很容易让外部的客户程序对其产生依赖,一旦产生依赖,接口就需要保持稳定。

3.6 优先使用对象组合,而不是类继承

  • 类继承通常为“白箱复用”,对象组合通常为“黑箱复用” 。
  • 继承在某种程度上破坏了封装性,子类父类耦合度高。
  • 而对象组合则只要求被组合的对象具有良好定义的接口,耦合度低。

更准确的继承关系,其实是一种类属关系,比如说动物继承自生物,人又继承自动物。很多时候我们会错用继承,通常我们会有另外一种表白方法:对象组合(class A中包含class B,B可以是指针或者对象)。

3.7 封装变化点

  • 使用封装来创建对象之间的分界层,让设计者可以在分界层的一侧进行修改,而不会对另一侧产生不良的影响,从而实现层次间的松耦合。

当设计的素养逐步提升,是可以在软件领域划分分界层,从传统面向对象思维,封装就是封装代码和数据,但是封装更高层次就是封装变化点,一侧变化,一侧稳定。

3.8 针对接口编程,而不是针对实现编程

  • 不将变量类型声明为某个特定的具体类,而是声明为某个接口。(针对业务类型,而不是像string等的具体类)
  • 客户程序无需获知对象的具体类型,只需要知道对象所具有的接口。
  • 减少系统中各部分的依赖关系,从而实现“高内聚、松耦合”的类型设计方案。

上篇中:

	vector<Line> lineVector;
	vector<Rect> rectVector;

就违背了针对接口编程的原则,而

vector<Shape*> shapeVector;  //放抽象接口
......
shapeVector[i]->Draw(e.Graphics);  //使用接口的通用方法即可

针对接口编程的原则通常是与依赖倒置原则(DIP)相辅相成,违背了一个,通常也就违背了另一个。

不同设计原则是从不同角度看问题的。

面向接口设计的深入思考:产业强盛的标准:接口标准化!

接口的标准化,也就代表了工作的分配

4. 将设计原则提升为设计经验

  • 设计习语 Design Idioms
    • Design Idioms 描述与特定编程语言相关的低层模式,技巧,惯用法。
  • 设计模式 Design Patterns
    • Design Patterns主要描述的是“类与相互通信的对象之间的组织关系,包括它们的角色、职责、协作方式等方面。
  • 架构模式 Architectural Patterns
    • Architectural Patterns描述系统中与基本结构组织关系密切的高层模式,包括子系统划分,职责,以及如何组织它们之间关系的规则。

下篇将会真正进入每一个模式,进入某一个模式之前,需要深入领会上面提到8大设计原则,只有这样才能灵活的运用某一种模式。

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

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

相关文章

C++面向对象编程(2)

目录 一. 问题引入 二. 右值引用 1. lvalue/rvalue/prvalue/xvalue 1.1 表达式与对象的概念 1.2 左值与右值 2. moving semantics 2.1 显示绑定 2.2 Move constructors 2.3 Move assignment operator 2.4 实例分析 // TODO Quiz REF 本章简单介绍下move语义的“来…

数据通信——传输层TCP(可靠传输原理的停止等待协议)

引言 基于上次的基本特性中的可靠传输特性&#xff0c;我们对其进行详细的分析。 对了&#xff0c;在讲解TCP前&#xff0c;先对缓存进行一个简单的描述。我们都听说过缓存这个东西&#xff0c;我们发来的数据&#xff0c;会暂时写入到缓存中进行预处理&#xff0c;随后从缓存中…

OpenAI发布ChatGPT企业级版本

本周一&#xff08;2023年8月28日&#xff09;OpenAI 推出了 ChatGPT Enterprise&#xff0c;这是它在 4 月份推出的以业务为中心的订阅服务。该公司表示&#xff0c;根据新计划&#xff0c;不会使用任何业务数据或对话来训练其人工智能模型。 “我们的模型不会从你的使用情况中…

最适合家用的洗地机哪个牌子好?好用洗地机推荐

在如今的快节奏下&#xff0c;许多人没有太多的时间去完成家务清洁这一大问题&#xff0c;但是随着洗地机的出现&#xff0c;很多小伙伴们都发现了这个智能清洁家电在我们的生活中频繁出现&#xff0c;比如最近比较火爆的智能洗地机&#xff0c;结合了吸拖洗为一体的高效清洁与…

Bigemap 在水土生态环境行业中的应用

工具 Bigemap gis office地图软件 BIGEMAP GIS Office-全能版 Bigemap APP_卫星地图APP_高清卫星地图APP 使用场景&#xff1a; 1. 土地利用占地管理&#xff1a; 核对数据&#xff0c;查看企业的实际占地是否超出宗地&#xff0c;污染面积。不方便现场勘测的地方&#xf…

Python实操GetOpenFilename打开文件选择框获取工作表信息

import xlwings as xw xlapp xw.App(visibleTrue, add_bookFalse) a xlapp.api.GetOpenFilename(Excel Files (*.xl*),*.xl*, 0, 0, 0, True) print(a[0]) slist [] for i in a:wk xw.Book(i)for s in wk.sheets:s_dict {}s_dict[工作簿名字] wk.names_dict[工作表名字] …

K 次取反后最大化的数组和【贪心算法】

1005 . K 次取反后最大化的数组和 给你一个整数数组 nums 和一个整数 k &#xff0c;按以下方法修改该数组&#xff1a; 选择某个下标 i 并将 nums[i] 替换为 -nums[i] 。 重复这个过程恰好 k 次。可以多次选择同一个下标 i 。 以这种方式修改数组后&#xff0c;返回数组 可能…

信看课堂笔记—LDO和DC-DC电路打PK

LDO&#xff08;low dropout voltage regulator&#xff0c;低压差线性稳压器&#xff09;和DC-DC(Direct current-Direct current converter&#xff0c;直流电压转直流电压转换器)电源是非常常见的电源电路&#xff0c;LDO 出来的比较早&#xff0c;像老戏骨一样&#xff0c;…

热烈祝贺天光长寿食品成功入选航天系统采购供应商库

经过航天系统采购平台的严审&#xff0c;单州天光长寿食品&#xff08;单县&#xff09;有限公司成功入选中国航天系统采购供应商库。航天系统采购平台是航天系统内企业采购专用平台&#xff0c;服务航天全球范围千亿采购需求&#xff0c;目前&#xff0c;已有华为、三一重工、…

邀您试用|一键生成企业数字人,深兰科技硅基大脑SaaS平台使用指南

深兰科技“硅基大脑SaaS平台”是在深兰科技硅基知识大模型的基础上进行领域知识和个性化人设等高维数据的定制化训练&#xff0c;并以其零代码、一键生成、低成本等特点&#xff0c;实现从形象到知识体系的全面自定义&#xff0c;满足企业在智能客服、数字员工和行业专家等多个…

学习乐趣无限:学乐多光屏P90助力儿童智能学习新纪元

在这个变革的浪潮中&#xff0c;学乐多光屏P90以其卓越的功能和深刻的教育理念&#xff0c;成为了智能儿童学习领域的引领者&#xff0c;为孩子们开启了全新的学习体验。 融合创新技术&#xff0c;引领学习变革 学乐多光屏P90凭借其独特的触摸和投影光学技术&#xff0c;为儿…

K8S访问控制------认证(authentication )、授权(authorization )、准入控制(admission control )体系

一、账号分类 在K8S体系中有两种账号类型:User accounts(用户账号),即针对human user的;Service accounts(服务账号),即针对pod的。这两种账号都可以访问 API server,都需要经历认证、授权、准入控制等步骤,相关逻辑图如下所示: 二、authentication (认证) 在…

1+X智慧安防系统实施与运维技能等级证产教融合基地建设方案

一、系统概述 1X智慧安防系统实施与运维技能等级证产教融合体系统融合了产业需求、教育培训和技能认证&#xff0c;通过课程培训、实训基地和实习实训等方式培养学员的技能水平&#xff0c;并通过技能认证来评估其能力&#xff0c;以满足智慧安防行业对人才的需求&#xff0c;并…

el-date-picker 等 点击无反应不回显问题解决

如上图&#xff0c;编辑回显正常&#xff0c;但是时间控件在拖动过程中时间不会跟随改变。 解决办法&#xff1a; <el-date-picker input"onInput()" ...><el-input input"onInput()" ...>js中onInput() {this.$forceUpdate();},

Approaching (Almost) Any Machine Learning Problem中译版

前言 Abhishek Thakur&#xff0c;很多kaggler对他都非常熟悉&#xff0c;2017年&#xff0c;他在 Linkedin 发表了一篇名为Approaching (Almost) Any Machine Learning Problem的文章&#xff0c;介绍他建立的一个自动的机器学习框架&#xff0c;几乎可以解决任何机器学习问题…

企业的固定资产管理怎么操作

一家拥有多台大型设备的工厂&#xff0c;这些设备需要定期进行保养和维护&#xff0c;以确保其正常运转。而企业内部员工由于专业知识和技能的不同&#xff0c;需要分工协作才能更好地完成各项工作任务。因此&#xff0c;在设备资产管理方面&#xff0c;如何实现高效、便捷、透…

Linux x86_64 C语言实现gdb断点机制

文章目录 前言一、trap指令简介二、调用ptrace三、创建breakpoints四、CONT 和 SINGLESTEP五、完整代码演示六、增加参数检测参考资料 前言 本文参考文章&#xff1a;Implementing breakpoints on x86 Linux 一、trap指令简介 将通过在断点地址向目标进程的内存中插入一条新…

【数据库必备插件】Navicat Premium 15安装教程

软件下载 软件&#xff1a;Navicat Premium版本&#xff1a;15语言&#xff1a;简体中文大小&#xff1a;68.85M安装环境&#xff1a;Win11/Win10/Win8/Win7硬件要求&#xff1a;CPU2.0GHz 内存4G(或更高&#xff09;下载通道①百度网盘丨64位下载链接&#xff1a;https://pan…

【MATLAB第70期】基于MATLAB的LightGbm(LGBM)梯度增强决策树多输入单输出分类预测模型(全网首发,敬请期待)

【MATLAB第70期】基于MATLAB的LightGbm(LGBM)梯度增强决策树多输入单输出分类预测模型&#xff08;全网首发&#xff0c;敬请期待&#xff09; (LGBM)是一种基于梯度增强决策树(GBDT)算法。 基于MATLAB的LightGbm即将研究测试上线。 下一个研究对象&#xff1a; ABCBOOST模型 一…

【大数据之Kafka】六、Kafka Broker工作流程

1 Zookeeper存储的Kafka消息 &#xff08;1&#xff09;启动zookeeper可客户端 [lyxhadoop102 zookeeper-3.5.7]$ bin/zkCli.sh&#xff08;2&#xff09;通过ls命令查看Kafka相关信息 [zk: localhost:2181(CONNECTED) 0] ls /kafka2 Kafka Broker总体工作流程 Zookeeper集…