依赖倒置原则(DIP)
1. 定义
(1)高层模块不应依赖于低层模块,两者都应该依赖于抽象。
(2)抽象不应该依赖于细节,细节应该依赖于抽象。
为什么是“倒置”这个词?
这是由于许多传统的软件开发方法,比如结构化分析和设计,总是倾向于创建一些高层依赖于低层模块、策略依赖于细节的软件结构。
实际上这些方法的目的之一就是要定义程序层次结构,该层次结构描述了高层模块怎样调用低层模块。
一个设计良好的面向对象的程序,其依赖于程序结构相对于传统的过程式方法设计的通常结构而言就是被“倒置”了。
高层模块包含了一个应用程序中的重要的策略选择和业务模块。正是这些高层模块才使得其所在的应用程序区别于其他。然而,如果这些高层模块依赖于低层模块,那么低层模块的改动就会直接影响到高层模块,从而迫使它们依次做出改动。
这种情况是非常荒谬的,本应该高层的策略设置模块去影响低层的细节实现模块。包含高层业务规则的模块应该优先并独立于包含实现细节的模块。无论如何高层模块都不应该依赖于低层模块。
2. 层次化
所有结构良好的面向对象架构都具有清晰的层次定义,每个层次通过一个定义良好的、受控的接口向外提供一组内聚的服务。

高层Policy使用低层的Mechanism,而Mechanism使用了更细节的Utility。这看起来似乎是正确的,然而存在一个隐伏的错误特征:
Policy对于其下一直到Utility的改动都是敏感的。这种依赖关系是传递的。Policy依赖于某些Utility层次。因此,Policy传递性地依赖于Utility。

上图展示了一个更为合适的模型,每个较高层次都为它所需要的服务声明一个抽象接口,较低的层次是吸纳了这些抽象接口,每个高层类都通过该抽象接口使用下一层,这样高层就不依赖于低层。低层反而依赖于在高层中声明的抽象服务接口。这不仅解除了Policy Layer对Utility Layer的传递依赖关系,甚至也解除了Policy Layer对于Mechanism Layer的依赖关系。
请注意这里的倒置不仅仅是依赖关系的倒置,它也是接口所有权的倒置。通常会认为工具库应该拥有自己的接口。但是当应用了DIP时,往往是客户拥有抽象接口,而它们的服务者则从这些抽象接口派生。
2.1 倒置的接口所有权
低层模块实现了在高层模块中声明并被高层模块调用的接口。
通过这种倒置的接口所有权,对于MechanismLayer或UtilityLayer的任何改动都不会再影响到PolicyLayer。而且,PolicyLayer可以在定义了符合PolicyServiceInterface的任何上下文中重用。这样,通过倒置这些依赖关系,创建了一个更灵活、更持久、更易改变的结构。
2.2 依赖于抽象
程序中所有的依赖关系都应该终止于抽象类或接口,更这个启发式规则,可知:
(1)任何变量都不应该持有一个指向具体类的指针或引用
(2)任何类都不应该从具体类派生
(3)任何方法都不应该覆写它的任何基类中的已经实现了的方法
当然,每个程序中都会有违反该启发规则的情况,有事必须要创建具体类的实例,而创建这些实例的模块将会依赖于它们。
此外,该启发规则对于那些虽是具体但却稳定的类来说似乎不太合理。如果一个具体类不太会改变,并且也不会创建其他类似的派生类,那么依赖于它并不会造成损害。
应用程序中所编写的大多数具体类都是不稳定的。我们不想依赖于这些不稳定的具体类。通过把它们隐藏在抽象接口的后面,可以隔离它们的不稳定性。
由客户类来声明它们需要的服务接口,那么仅当客户需要时才会对接口进行改变。这样,改变实现抽象接口的类就不会影响到客户。
3. 一个简单的例子
依赖倒置可以应用于任何存在一个类向另一个类发送消息的地方。
例如:Button对象和Lamp对象之间的情形。
Button对象感知外部环境的变化。当接收到Poll消息时,它会判断是否被用户“按下”。它不关心是通过什么样的机制去感知的。如何设计一个用Button对象控制Lamp对象的系统呢?

上图展示了一个不成熟的设计。Button对象接收Poll消息,判断按钮是否被按下,接着简单地发送turnOn或turnOff消息给Lamp对象。Button类直接依赖于Lamp类。这个依赖关系意味着当Lamp类改变时,Button类会受到影响。此外,想要重用Button来控制一个Motor对象是不可能的。在这个设计中,Button控制着Lamp对象,并且也只能控制Lamp对象。
class Button {
Lamp itsLamp;
void poll() {
itsLamp.turnOn();
}
}
这个方案违反了DIP,应用程序的高层策略没有和低层实现分离。抽象没有和具体细节分离。没有这种分离,高层策略就自动地依赖于低层模块,抽象就自动地依赖于具体细节。
3.1 找到潜在的抽象
什么是高层策略呢?它是应用背后的抽象,是那些不随具体细节的改变而改变的真理。
在Button/Lamp例子中,背后的抽象是检测用户的开关并将指令传给目标对象。
用什么机制检测用户的指令呢?无关紧要,目标对象是什么?同样无关紧要,这些都是不会影响到抽象的具体细
通过倒置对Lamp对象的依赖关系,可以改进设计。

可以看出Button现在和一个ButtonServer的接口关联起来。ButtonServer接口提供了一些抽象方法,Button可以使用这些方法来开启或关掉一些东西。Lamp实现了ButtonServer接口。这样,Lamp现在是依赖于别的东西了,而不是被依赖了。
设计可以使Button控制哪些愿意实现ButtonServer接口的任何设备,这赋予我们极大的灵活性。同时也意味着Button对象将能够控制还没有被创造出来的对象。
4. 总结
使用传统的过程化程序设计所创建出来的依赖关系结构,策略是依赖于细节的。只是糟糕的,因为这样会使策略收到细节改变的影响。面向对象的程序设计导致了依赖关系结构,使得细节和策略都依赖于抽象,并且常常是客户用于服务接口
事实上,这种依赖关系的倒置正是好的面向对象设计的标志所在。使用何种语言来编写程序时无关紧要的。如果程序的依赖关系是倒置的,它就是面向对象的设计。如果程序的依赖关系不是倒置的,它就是过程化的设计。
依赖倒置原则是实现许多面向对象技术所宣称的好处的基本低层机制。它的正确应用对于创建可重用的框架来说是必须的。同时它对于构建在变化面前富有弹性的代码也是非常重要的。由于抽象和细节被彼此距离,所以代码也非常容易维护。
依赖倒置原则(DIP)的更多相关文章
- C#软件设计——小话设计模式原则之:依赖倒置原则DIP
前言:很久之前就想动笔总结下关于软件设计的一些原则,或者说是设计模式的一些原则,奈何被各种bootstrap组件所吸引,一直抽不开身.群里面有朋友问博主是否改行做前端了,呵呵,其实博主是想做“全战”, ...
- 7.12 其他面向对象设计原则3: 依赖倒置原则DIP
其他面向对象设计原则3: 依赖倒置原则DIP The Dependency Inversion Principle7.1 依赖倒置原则DIP The Dependency Inversion Pr ...
- 依赖倒置原则DIP&控制反转IOC&依赖注入DI
依赖倒置原则DIP是软件设计里一个重要的设计思想,它规定上层不依赖下层而是共同依赖抽象接口,通常可以是上层提供接口,然后下层实现接口,上下层之间通过接口完全透明交互.这样的好处,上层不会因依赖的下层修 ...
- 对依赖倒置原则(DIP)及Ioc、DI、Ioc容器的一些理解
1.概述 所谓依赖倒置原则(Dependence Inversion Principle)就是要依赖于抽象,不要依赖于具体.简单的说就是要求对抽象进行编程,不要对实现进行编程,这样就降低了客户与实现模 ...
- 对依赖倒置原则(DIP)及Ioc、DI、Ioc容器的一些理解(转)
所谓依赖倒置原则(Dependence Inversion Principle)就是要依赖于抽象,不要依赖于具体.简单的说就是要求对抽象进行编程,不要对实现进行编程,这样就降低了客户与实现模块间的耦合 ...
- 依赖倒置原则(DIP)
什么是依赖倒置呢?简单地讲就是将依赖关系倒置为依赖接口,具体概念如下: 1.上层模块不应该依赖于下层模块,它们共同依赖于一个抽象(父类不能依赖子类,它们都要依赖于抽象类) 2.抽象不能依赖于具体,具体 ...
- 设计模式学习--面向对象的5条设计原则之依赖倒置原则--DIP
一.DIP简介(DIP--Dependency Inversion Principle): 1.高层模块不应该依赖于低层模块,二者都应该依赖于抽象.2.抽象不应该依赖于细节,细节应该依赖于抽象. ...
- 深入理解JavaScript系列(22):S.O.L.I.D五大原则之依赖倒置原则DIP
前言 本章我们要讲解的是S.O.L.I.D五大原则JavaScript语言实现的第5篇,依赖倒置原则LSP(The Dependency Inversion Principle ). 英文原文:htt ...
- IOS设计模式的六大设计原则之依赖倒置原则(DIP,Dependence Inversion Principle)
定义 高层模块不应该依赖于低层模块,二者都应该依赖于抽象:抽象不应该依赖细节:细节应该依赖抽象. 定义解读 依赖倒置原则在程序编码中经常运用,其核心思想就是面向接口编程,高层模块不应该依赖低层模块(原 ...
- 依赖倒置原则DIP(面向接口编程—OOD)
含义: 1.高层模块不应该依赖底层模块,两者都应该依赖其抽象. 2.抽象不应该依赖细节. 3.细节应该依赖抽象. 底层模块:不可分割的原子逻辑. 高层模块: 原子逻辑的再组装. 抽象:接口或者抽象类, ...
随机推荐
- Java 之 数据库连接池
一.数据库连接池 1.连接池概念 连接池其实就是一个容器(集合),存放数据库连接的容器. 当系统初始化好后,容器被创建,容器中会申请一些连接对象,当用户来访问数据库时,从容器中获取连接对象,用户访问之 ...
- Tomcat运行一段时间后,自动停止关闭,To prevent a memory leak,Druid 数据库连接自动关闭, the JDBC Driver has been forcibly unregistered.
1. Tomcat 错误日志 tail -100f tomcat9/logs/catalina.out 21-Sep-2017 23:05:39.301 INFO [Thread-5] org.apa ...
- JDK的安装(mac)
1.第一步安装brew 教学网址 2.用brew安装jdk. brew update brew cask install java(未翻墙时长很长,大概猴年马月两个小时) 安装完成后就可以执行JAVA ...
- 【转】TI DSP C6657学习之——编译静态库.lib
熟悉C++开发的的小伙伴都知道,我们一般代码中往往要引入许多第三方编译好的库,有些是静态链接库static library, 有些是动态链接库dll.引入库的目的一是减少代码的编译时间,二是只提供函数 ...
- PHP标记风格,编码规范
PHP标记风格PHP一共支持4种标记风格 <?php echo "这是XML风格的标记"; ?> 脚本风格 <script language="php& ...
- 绘图 Matplotlib Numpy Pandas
丈夫气力全,一个拟当千.猛气冲心出,视死亦如眠. 绘图 Matplotlib可视化是在整个数据挖掘的关键辅助工具,可以清晰的理解数据,从而调整我们的分析方法. 能将数据进行可视化,更直观的呈现使数据更 ...
- JQuery EasyUI treegrid展开与折叠,以及数据加载两次的问题
问题:做项目的时候遇到代码生成的页面,只默认展开了一级节点,每次操作之后刷新还要手动一级一级展开,太麻烦了 官方API:http://www.jeasyui.net/plugins/186.html ...
- linux网卡名称修改
vim /etc/sysconfig/grub ,在倒数第二行添加如下代码 net.ifnames=0 biosdevname=0 GRUB_TIMEOUT=5 GRUB_DISTRIBUTOR=&q ...
- 云计算---记一次黑客攻击openstack创建的虚拟机
一:问题定位 现象: 近期发现有几台openstack云主机被修改密码并被肉鸡. 黑客操作日志: -- :: ##### root tty1 : #### -- :: top -- :: ##### ...
- 如何利用AI识别未知——加入未知类(不太靠谱),检测待识别数据和已知样本数据的匹配程度(例如使用CNN降维,再用knn类似距离来实现),将问题转化为特征搜索问题而非决策问题,使用HTM算法(记忆+模式匹配预测就是智能),GAN异常检测,RBF
https://www.researchgate.net/post/How_to_determine_unknown_class_using_neural_network 里面有讨论,说是用rbf神经 ...