工作这么久才明白的SOLID设计原则_第1页
工作这么久才明白的SOLID设计原则_第2页
工作这么久才明白的SOLID设计原则_第3页
工作这么久才明白的SOLID设计原则_第4页
工作这么久才明白的SOLID设计原则_第5页
已阅读5页,还剩4页未读 继续免费阅读

下载本文档

版权说明:本文档由用户提供并上传,收益归属内容提供方,若内容存在侵权,请进行举报或认领

文档简介

前序做C语言开发的应该都知道,C是面向过程开发的,而c++是面向对象开发的。而封装、继承与多态是面向对象开发的三大特征。但你可能不知道OOD(Object-OrientedDesign)还有五大基本原则,被Bob大叔称为SOLID原则,字母为每个原则的首字母,遵循这些原则能够让你的代码在扩展性、维护性以及重用性提高。而这些不正是我们所追求的吗?接下来我们就一块学习学习这些原则,内容较多,建议先收藏后反复观看,文章末尾有很多参考链接。五大基本原则-SOLID1.SRPSRP(TheSingleResponsibilityPrinciple)单一职责原则。SRP是SOLID五大设计原则中最容易被误解的一个,SRP不就是每个模块都应该只做一件事吗?非也非也,这只是在实现底层细节的实际原则,并非是SRP的全部。SRP最初是这样描述的:任何一个模块都应该有且只有一个被修改的原因(Thereshouldneverbemorethanonereasonforaclasstochange)SRP的定义几经迭代,最终被RobertC.Martin在《CleanArchitecture》中定义为:任何一个软件模块都应该只对某一类行为者负责那么上文中提到的软件模块究竟是指什么呢?大部分情况下,其最简单的定义就是指一个源代码文件。然而有些编程语言和编程环境并不是用源代码文件来存储程序的。在这些情况下,软件模块指的就是一组紧密相关的函数和数据结构。来看一个正面例子,C标准库中的<math.h>模块就是用来处理数学相关的,<string.h>模块就是用来处理字符串相关的的,等等...,每个模块职责都比较明确。再来看一个反面例子,还记得刚开始学习单片机编程的时候,项目工程中从始至终就一个main源文件,真的是连头文件都不写的,一个main里面包含了LED、按键等相关全部代码,这明显是不符合SRP这一原则的。就像下图一样。2.OCPOCP(TheOpen-ClosedPrinciple)开放封闭原则。OCP的定义如下:软件实体应当对扩展开放,对修改关闭(Softwareentitiesshouldbeopenforextension,butclosedformodification)设计良好的计算机软件应该易于扩展,同时抗拒修改。换句话说,一个良好的计算机系统应该在不需要修改的前提下就可以轻易被扩展。OCP是系统框架设计的主导原则,其主要目的是让系统易于扩展,同时限制其每次被修改所影响的范围。实现的方式是通过将系统划分为一系列的组件,并且将这些组件的依赖关系按层次结构进行组织,使得高阶的组件不会因低阶组件被修改而受到影响。我们来看一个网络中协议的例子,如下图:图中分为三个层级,最上方层级最高,每个层级有若干组件,高层级的组件依赖低层级的组件。3.LSPLSP(LSP-LiskovSubstitutionPrinciple)里氏替换原则。LSP定义:使用基类对象指针或引用的函数必须能够在不了解衍生类的条件下使用衍生类的对象(Functionsthatusepointersorreferencestobaseclassesmustbeabletouseobjectsofderivedclasseswithoutknowingit)里氏代换原则中说,任何基类可以出现的地方,子类一定可以出现。只有当衍生类可以替换掉基类,软件单位的功能不受到影响时,基类才能真正被复用,而衍生类也能够在基类的基础上增加新的行为。我们来看一个经典的反面案例,正方形(Square)与长方形(Rectangle)。那这里提出疑问了,正方形和长方形之间的继承关系是如何呢?如果A

is-a

B,则认为B是基类,A为子类,A应该继承B。那这个简单了,众所周知正方形是长方形,立即推,长方形是基类,正方形是子类,正方形应该继承长方形。基类长方形中有SetHeight()和SetWidth()方法,但是子类正方形有一个特点,长和宽是相等的,所以在实现SetHeight()和SetWidth()都必须同时设置宽和高,我们来看代码。还是那句话任何基类可以出现的地方,子类一定可以出现,所以在创建Rectangle对象指针的时候,这里其实给的是子类的对象,最终代码运行断言报错。Assertion

failed:

rect->GetArea()

==

10,

file

.\main.cpp,

line

10如果new出来的对象是Rectangle,则程序能够正常执行。如果new出来的是Square则会进入断言。从而Square不能代替Rectangle,所以不符合LSP原则,实际上Square并不是Rectangle的子类。正方形是长方形,但是他们的行为并不一样,所谓的行为,就是抽象出来的东西。正方形只要有一个设置边长的方法就行了,而长方形需要设置宽和高两种方法。4.ISPISP(ISP-InterfaceSegregationPrinciple)接口隔离原则。ISP定义:不应强制客户端依赖于它们不使用的接口(Clientsshouldnotbeforcedtodependuponinterfacesthattheydonotuse.)该原则还有另外一个定义:一个类对另一个类的依赖应该建立在最小的接口上(Thedependencyofoneclasstoanotheroneshoulddependonthesmallestpossibleinterface)假如现在有一个OPS类。用户1只需要使用OPS类的op1方法,用户2只需要OPS类的op2方法,但是呢,OPS类除了提供op1和op2方法还提供了若干方法。如下图:此时用户2没什么意见,心想着反正能实现我要的功能就可以了。但是呢,用户1不愿意,于是就去找开发人员理论,我就要实现一个op1功能,给我整这么多依赖干啥。除了这个功能,其他的全部都给我隐藏掉,下班之前我就要,说完头一扭就走了。开发人员心想,这么简单的事情,让我下班之前给你,这不是眼看人低吗,说完就在用户1和OPS之间又封装一层IU1Ops接口,两分钟搞定。于是就去跟用户1说,你用IU1Ops,里面有你要的接口,拿去用吧。说完头一扭就走了。模型如下图:接口隔离原则和单一职责都是为了提高类的内聚性、降低它们之间的耦合性,体现了封装的思想,但两者是不同的:单一职责原则注重的是职责,而接口隔离原则注重的是对接口依赖的隔离。单一职责原则主要是约束类,它针对的是程序中的实现和细节;接口隔离原则主要约束接口,主要针对抽象和程序整体框架的构建。5.DIPDIP(DIP-DependencyInversionPrinciple)依赖倒置原则。DIP定义:高层次的模块不应该依赖低层次的模块,他们都应该依赖于抽象(Highlevelmodulesshouldnotdependuponlowlevelmodules.Bothshoulddependuponabstractions)抽象不应该依赖于具体实现,具体实现应该依赖于抽象(Abstractionsshouldnotdependupondetails.Detailsshoulddependuponabstractions)这个名字看着有点别扭,“依赖”还“倒置”,这到底是啥意思?依赖指两个相对独立的对象,当一个对象负责构造另一个对象的实例,或者依赖另一个对象的服务时,这两个对象之间主要体现为依赖关系。老样子,看示例,现在甲方需要能让BMW车跑起来的功能,再看一下乙方的设计。司机有驾驶BMW车辆的方法,BMW车辆有run的方法,所以是甲方满足需求的。后来甲方需求变了,甲方不仅要能开BMW车,也要能开Benz车,却发现使用乙方原来的的设计Benz车却开不起来,因为+driver()只接受BMW的车辆,不接受Benz的车辆,这明显不合理,一个司机会开BMW却不会开Benz,太反常了。于是去找乙方重新设计。乙方也认识到了不足,经过多方讨论终于有了如下设计。对于每种车辆,都应该有一个run的方法,所有的车辆都应该继承ICar实现。而司机呢,也不应该依赖具体的车辆,而应该依赖所有车辆的抽象方法。原本直接指向BMW的依赖箭头被反置了。现在即使甲方在添加别的车辆,司机仍然能开。不会出现只会开BMW而不会开其他车辆的情况了。简单来说,依赖倒转原则就是指:代码要依赖于抽象的类,而不要依赖于具体的类

温馨提示

  • 1. 本站所有资源如无特殊说明,都需要本地电脑安装OFFICE2007和PDF阅读器。图纸软件为CAD,CAXA,PROE,UG,SolidWorks等.压缩文件请下载最新的WinRAR软件解压。
  • 2. 本站的文档不包含任何第三方提供的附件图纸等,如果需要附件,请联系上传者。文件的所有权益归上传用户所有。
  • 3. 本站RAR压缩包中若带图纸,网页内容里面会有图纸预览,若没有图纸预览就没有图纸。
  • 4. 未经权益所有人同意不得将文件中的内容挪作商业或盈利用途。
  • 5. 人人文库网仅提供信息存储空间,仅对用户上传内容的表现方式做保护处理,对用户上传分享的文档内容本身不做任何修改或编辑,并不能对任何下载内容负责。
  • 6. 下载文件中如有侵权或不适当内容,请与我们联系,我们立即纠正。
  • 7. 本站不保证下载资源的准确性、安全性和完整性, 同时也不承担用户因使用这些下载资源对自己和他人造成任何形式的伤害或损失。

评论

0/150

提交评论