软件设计的基本原则如下:
1、单一职责原则(SRP):一个类或模块应该只负责一项任务或功能。
2、开闭原则(OCP):软件实体(类、模块、函数等)应该对扩展开放,对修改关闭。
3、里氏替换原则(LSP):子类应该能够替换其父类并且不会破坏程序的正确性。
4、接口隔离原则(ISP):客户端不应该强制依赖它不需要的接口,即应该将接口拆分成更小的部分。陆巧
5、依赖倒置原则(DIP):高层模块不应该依赖于底层模块,它们都应该依赖于抽象接口。
6、最少知识原则(LKP):一个软件实体应该尽可能少地了解其他实体的存在,尽量降低它们之间的耦合。
7、组合/聚合复用原则(CARP):尽可能使用组合/聚合来实现代码复用,而不是继承。
8、优先使用简单的设计原则(KISS):在设计时,应该尽可能简单明了,不要过度设计。
9、开发封闭原则(ADP):模块间的依赖关系应该早掘键通过抽象来建立,而不是具体的实现类。
10、信息专注原则(ISP):在一个模块或类中,只应该包含与其相关的信息,而不是与之无关的信息。
开闭原则的作用:
开闭原则是面向对象程序设计的终极目标,它使软件实体拥有一定的适应性和灵活性的同时具备稳定性和延续性。具体来说,其作用如下。
(1)对软件测试的影响:软件遵守开闭原则的话,软件测试时只需要对扩展的代码进行测试就可以了,因为原有的测试代码仍然能够正常运行。
(2)可以提高代码的可复用性:粒度越小,被复用的可能性就越大;在面向对象的程序设计中,根据原子和抽象编程可以提高代码的可复用性。
(3)可以提高软件的可维护性:遵守开闭原则的软件散滚,其稳定性高和延续性强,从而易于扩展和维护。
kiss,在英语里是察卜前亲吻的意思,既可以作为名词,也可以作为动词。 但它还有另外一个意思,一个缩写:Keep It Simple and Stupid,保持简单浅显的意思。 能够将复杂的事情做得非常简单,使用起来非常简单自然,还是需要不少功力的。如此看来,kiss原则弊档在很多方面都大有用处。程序设计风格可以KISS, 家庭装修可以KISS, 美术设计可以KISS, 界面设计也可以KISS,写论文也要kiss。从客户角度考虑,保证客户的使用简单,需要kiss原则,从框架设计考虑,可败清以让复杂的问题简单的表现和处理。作者:oec2003
公众号:不止dotNET
本文继续来介绍接口隔离原则(ISP)和依赖倒置原则(DIP),这两个原则都和接口和继承有关。文章最后会简单介绍几个除了 SOLID 原则之外的原则。
提起接口,开发人员的第一反应可能是面向对象编程语言中的 interface ,但接口更广义的理解会包含:
不管是上面的哪一种,要想设计好,就需要用到接口隔离原则了。
接口隔离原则的定义是:
接口被设计出来后,就会有地方对接口进行调用,调用的地方希望接口中提供的方法都是他需要的,所以在接口设计的时候,需要考虑应该将哪些方法放入其中,让调用者使用,这就是对定义的解释。
相反,如果不精心设计,接口就会变得越来越庞大,会带来两个问题:
1、在一个更高层的接口中添加一个方法只是为了某一个子类使用,所有的子类都必须对其实现,或提供一个默认实现;
2、接口中包罗万象,调用者可能会误用其中的方法。
举个例子:我们现在正在开发 SaaS 产品,里面会涉及到对租户的 *** 作,比如租户需要注册、登录等,抽象成接口代码如下:
上面的 *** 作是针对租户这个角色的,现在有新的需求来了,对于 SaaS 厂商的管理员来说,希望能禁用租户,一种偷懒的做法就是直接在 ITenant 接口中添加禁用的方法,如下:
上面的代码就违反了接口隔离原则,因为在普通租户的使用场景下,并不希望能调用到 Diabled 方法,正确的做法是将这个方法抽象到一个新的接口中,如下:
可以看出来,改造之后,每个接口的职责更加单一了,好像跟单一职责有点类似,仔细想想,还是有些区别,单一职责原则针对的是方法、类兆扒和接口的设计。而接口隔离原则更侧重于接口的设计告做,另一方面就是思考的角度不同,在上面例子中,按照普通租户和管理员两种不同角色的维度来思考并进行拆分。
这个原则的名字中有两个关键词「依赖」和「倒置」,先来看看这两个词是什么意思?
依赖:在面向对象的语言中,所说的依赖通常指类与类之间的关系,比如有个用户类 User 和日志类 Log , 在 User 类中需要记录日志,就需要引入日志类 Log,这样 User 类就对 Log 类产生了依赖,代码如下:
倒置:有依赖的倒置,那肯定就有正常的依赖,我们正常的编程思维都是从上而下来编写业务逻辑的,遇到分支就写 if ,遇到循环就写 for ,需要创建对象就 new 一个,就像上面的代码,上面的代码就是一种正常的依赖。User 类依赖了 Log 类,如果倒置了,那就是 User 类不再依赖 Log 类了,下面会进一步来解释。
正常的依赖会带来的问题是:User 类和 Log 类高度耦合,当有一天我们想使用 NLog 或者 Serilog 替换 Log 类时,就需要改动 User 类,说明日志类的实现是不稳定的,而依赖一个不稳定的东西,从架构设计的角度来看,不是一个好的做法。解决此问题就需要用到依赖倒置原则。
先来看看依赖倒置原则的定义:
什么是高层模块?什么是低层模块?按照上面的代码示例,User 类是高层模块,Log 类是低层模块,二者都要依赖于抽象,就需要提取接口了:
调整后的代码 User 类中依赖变成了 ILog 接口,日志的实现类 Log 也依赖 ILog 接口,即从 ILog 接口继承而来,现在都是依赖 ILog 接口,这就是依赖倒置。
当想要将日志组件替换为 NLog 时,只需要创建一个新的类 NLogAdapter 类继承 ILog 接口,在 NLogAdapter 类中引入 NLog 组件。
这样,当日志组件替换的时候,User 类就不用修改了,因为 User 类的构造函数中使用的是 ILog 接口来接收的日志组件的对象,那到底是谁决定传递 Log 对象还是 NLogAdapter 对象呢?这就要引入一个新的概念叫「依赖注入」。
关于依赖注入可以看我之前写的两篇文章:
依赖倒置是一种架构设计思想,指导架构层面的设计,依赖注族友昌入则是一种具体的编码技巧,用来实现这种设计思想。
除了 SOLID 五大原则之外,还有一些原则也在指引我们设计好的代码架构方面发挥着作用:
KISS 的全称是:Simple and Stupid ,该原则就是告诉我们,在设计时要尽量保持简单,大道至简嘛。这里的简单不完全是指代码的简洁。现在已经不是单打独斗的时代,大部分情况下开发人员都是在一个团队中协同工作,所以我认为对简单的理解可以分为:
将复杂的东西能够深入浅出,做到简单、简洁,这是能力的体现。
YAGNI 的全称是:You Ain’t Gonna Need It。直译就是:你不会需要它。核心思想就是指导我们不要做过度设计。
1、当我们能识别到代码的变化点的时候,可以预留扩展点,但不要提前做复杂的实现;
2、持续重构来优化代码,而不是一开始就提取各种通用方法,例如一个私有函数只有一个调用的时候,就放在类里面,离调用者最近的地方,当有不止一处都会使用时,再考虑重构来进行通用方法的抽取。
过度设计会浪费资源,让代码复杂度变大,难以阅读和维护。
DRY 的全称是:Don’t Repeat Yourself ,就是不要重复自己,提升代码的复用性,告别 CV 大法。
很多初级程序员都喜欢面向 Ctrl+C、Ctrl+V 编程,当需求变化的时候,很容易就遗漏一些场景,但即便是复制粘贴也不完全都是违反 DRY 。
代码的重复有两种情况:
1、代码的逻辑重复,语义也重复:这种违反了 DRY ,需要进行重构;
2、代码的逻辑重复,语义不重复:在某个阶段,两段代码逻辑是相同的,但其实是两种不同的应用场景,语义不一样,就没有违反 DRY。如果对这种代码进行重构提取成公共方法,随着业务发展,两种不同的场景独立演化了,稍不注意,代码中就会出现各种 if 判断,影响可读性和可维护性。
LOD 全称是:The Least Knowledge Principle ,也被称之为迪米特法则。该法则有两条指导原则:
1、不该有直接依赖关系的类之间,不要有依赖;
2、有依赖关系的类之间,尽量只依赖必要的接口。
其实就是一直流传的代码要高内聚、低耦合,单一职责和接口隔离想要表达的也是这个意思,区别只是侧重点有所不同:
各种原则之间相辅相成,有很多只是有些细微的差别,慢慢理解原理,才能以不变应万变。
欢迎分享,转载请注明来源:内存溢出
评论列表(0条)