问题由来

  问题由来:类A直接依赖类B,假如要将类A改为依赖类C,则必须通过修改类A的代码来达成。这种场景下,类A一般是高层模块,负责复杂的业务逻辑;类B和类C是低层模块,负责基本的原子操作;假如修改类A,会给程序带来不必要的风险。
解决方案:将类A修改为依赖接口I,类B和类C各自实现接口I,类A通过接口I间接与类B或者类C发生联系,则会大大降低修改类A的几率。
  依赖倒置原则基于这样一个事实:相对于细节的多变性,抽象的东西要稳定的多。以抽象为基础搭建起来的架构比以细节为基础搭建起来的架构要稳定的多。在java中,抽象指的是接口或者抽象类,细节就是具体的实现类,使用接口或者抽象类的目的是制定好规范和契约,而不去涉及任何具体的操作,把展现细节的任务交给他们的实现类去完成。
  依赖倒置原则的核心就是要我们面向接口编程,理解了面向接口编程,也就理解了依赖倒置。

定义

抽象不应该依赖于细节,细节应当依赖于抽象。换言之,要针对接口编程,而不是针对实现编程。

3层含义:
1)高层模块不应该依赖低层模块,两者都应该依赖其抽象。
2)抽象不应该依赖细节。
3)细节应该依赖抽象。

依赖倒置原则在java语言中的表现:
1)模块间的依赖通过抽象发生,实现类之间不发生直接的依赖关系,其依赖关系是通过接口或抽象类产生的。即使用接口和抽象类进行变量类型声明、参数类型声明、方法返回类型声明,以及数据类型的转换等,而不要用具体类来做这些事情。
2)接口或抽象类不依赖于实现类。
3)实现类依赖接口或抽象类。
即:面向接口编程。
4)在实现依赖倒置原则时,需要针对抽象层编程,而将具体类的对象通过依赖注入的方式注入到其他对象中。
5)在大多数情况下,开闭原则、里氏替换原则、依赖倒置原则会同时出现,开闭原则是目标,里氏替换原则是基础,依赖倒置是手段,他们相辅相成,互相补充,目标一致,只是分析问题时所站的角度不同而已。

依赖倒置原则最佳实践

依赖倒置原则的本质是通过抽象(接口或抽象类)使各个类或模块的实现彼此独立,不互相影响,实现木块间的松耦合,我们怎么在项目中使用这个规则呢?只要遵循以下几个规则即可:

1.每个类尽量都有接口或抽象类,或者抽象类和接口两者都具备
这是依赖倒置的基本要求,接口和抽象类都是属于抽象的,有了抽象才可能依赖倒置

2.变量的表面类型尽量是接口或者抽象类

3.任何类都不应该从具体类派生
如果一个项目处于开发状态,确实不应该有从具体类派生出子类的情况,但是这也不是绝对的,因为人是会犯错误的,有时设计缺陷是在所难免的,因此只要不超过两层的继承都是可以忍受的,特别是负责项目维护的同志,基本上可以不考虑这个规则,为什么呢?维护工作基本上都是扩展开发,修复行为。

4.尽量不要覆写基类的方法
如果基类是一个抽象类,而且这个方法已经实现了,子类尽量不要覆写。

5.结合里氏替换原则使用
结合里氏替换原则,我们可以得出一个通俗的规则:接口负责定义public属性和方法,并且声明与其他对象的依赖关系,抽象类负责公共构造部分的实现,实现类准确的实现业务逻辑,同时在适当的时候对父类进行细化。
什么是倒置呢?依赖正置就是类间的依赖是实实在在的实现类间的依赖,也就是面向实现编程,这也是正常人的思维。我们根据系统设计的需要产生了抽象间的依赖,代替了人们传统思维中的事物间的依赖,倒置就是从这里产生的。就是说与我们一般的思维不同就是倒置。

6.依赖倒置原则的是6个设计原则中最难以实现的原则,它是实现开闭原则的重要途径,依赖倒置原则没有实现,就别想实现对扩展开放,对修改关闭。在项目中,大家只要记住是“面向接口编程”就基本上抓住了依赖倒置原则的核心。

例子

  依赖倒置原则的核心思想是面向接口编程,我们依旧用一个例子来说明面向接口编程比相对于面向实现编程好在什么地方。场景是这样的,母亲给孩子讲故事,只要给她一本书,她就可以照着书给孩子讲故事了。代码如下:

class Book{  public String getContent(){  return "很久很久以前有一个阿拉伯的故事……";  }  }  class Mother{  public void narrate(Book book){  System.out.println("妈妈开始讲故事");  System.out.println(book.getContent());  }  }  public class Client{  public static void main(String[] args){  Mother mother = new Mother();  mother.narrate(new Book());  }  }  

运行结果:
妈妈开始讲故事
很久很久以前有一个阿拉伯的故事……

运行良好,假如有一天,需求变成这样:不是给书而是给一份报纸,让这位母亲讲一下报纸上的故事,报纸的代码如下:

    class Newspaper{  public String getContent(){  return "林书豪38+7领导尼克斯击败湖人……";  }  }  

这位母亲却办不到,因为她居然不会读报纸上的故事,这太荒唐了,只是将书换成报纸,居然必须要修改Mother才能读。假如以后需求换成杂志呢?换成网页呢?还要不断地修改Mother,这显然不是好的设计。原因就是Mother与Book之间的耦合性太高了,必须降低他们之间的耦合度才行。
我们引入一个抽象的接口IReader。读物,只要是带字的都属于读物:

interface IReader{  public String getContent();  }  

Mother类与接口IReader发生依赖关系,而Book和Newspaper都属于读物的范畴,他们各自都去实现IReader接口,这样就符合依赖倒置原则了,代码修改为:

    class Newspaper implements IReader {  public String getContent(){  return "林书豪17+9助尼克斯击败老鹰……";  }  }  class Book implements IReader{  public String getContent(){  return "很久很久以前有一个阿拉伯的故事……";  }  }  class Mother{  public void narrate(IReader reader){  System.out.println("妈妈开始讲故事");  System.out.println(reader.getContent());  }  }  public class Client{  public static void main(String[] args){  Mother mother = new Mother();  mother.narrate(new Book());  mother.narrate(new Newspaper());  }  } 

运行结果:
妈妈开始讲故事
很久很久以前有一个阿拉伯的故事……
妈妈开始讲故事
林书豪17+9助尼克斯击败老鹰……

这样修改后,无论以后怎样扩展Client类,都不需要再修改Mother类了。这只是一个简单的例子,实际情况中,代表高层模块的Mother类将负责完成主要的业务逻辑,一旦需要对它进行修改,引入错误的风险极大。所以遵循依赖倒置原则可以降低类之间的耦合性,提高系统的稳定性,降低修改程序造成的风险。

采用依赖倒置原则给多人并行开发带来了极大的便利,比如上例中,原本Mother类与Book类直接耦合时,Mother类必须等Book类编码完成后才可以进行编码,因为Mother类依赖于Book类。修改后的程序则可以同时开工,互不影响,因为Mother与Book类一点关系也没有。参与协作开发的人越多、项目越庞大,采用依赖导致原则的意义就越重大。现在很流行的TDD开发模式就是依赖倒置原则最成功的应用。

设计模式系列:依赖倒置原则相关推荐

  1. 胖虎白话学习设计模式之依赖倒置原则(Dependence Inversion Principle)

    胖虎白话学习设计模式之依赖倒置原则(Dependence Inversion Principle) 记录胖虎学习设计模式过程,不许勿喷,转载请注明出处! (此博文为胖虎在PDF上截取.觉得写得通俗易懂 ...

  2. JAVA【设计模式】依赖倒置原则

    依赖倒置原则 一.设计模式的规范 二.依赖倒置原则 三.示例 非依赖倒置原则(硬编码) 依赖倒置原则 UML关系图 一.设计模式的规范 设计模式遵循六⼤原则:单⼀职责( ⼀个类和⽅法只做⼀件事 ).⾥ ...

  3. ASP.NET 设计模式中依赖倒置原则

    依赖倒置原则 A.高层次的模块不应该依赖于低层次的模块,他们都应该依赖于抽象. B.抽象不应该依赖于具体,具体应该依赖于抽象. 依赖倒置原则 A.高层次的模块不应该依赖于低层次的模块,他们都应该依赖于 ...

  4. 设计模式:依赖倒置原则

    依赖倒置原则定义 依赖倒置原则(Dependence Inversion Principle ,DIP)定义如下: High level modules should not depend upon ...

  5. java 依赖倒置_设计模式之三依赖倒置原则(DIP)

    依赖倒置(Dependence Inversion Principle,DIP) High level modules should not deppend oupon low level modul ...

  6. 嘻哈说:设计模式之依赖倒置原则

    1.定义 按照惯例,首先我们来看一下依赖倒置原则的定义. 抽象不应该依赖于细节,细节应当依赖于抽象. 换言之,要针对接口编程,而不是针对实现编程. 为什么要这样说呢? 因为细节具有易变性,非常的不稳定 ...

  7. 设计原则(单一职责原则 开放封闭原则 里氏替换原则 依赖倒置原则 接口隔离原则 迪米特法则)

    设计原则 单一职责原则(SRP) 从三大特性角度看原则: 应用的设计模式: 开放封闭原则(OCP) 从三大特性角度看原则: 应用的设计模式: 里氏替换原则(LSP) 从三大特性角度看原则: 应用的设计 ...

  8. C#软件设计——小话设计模式原则之:依赖倒置原则DIP

    前言:很久之前就想动笔总结下关于软件设计的一些原则,或者说是设计模式的一些原则,奈何被各种bootstrap组件所吸引,一直抽不开身.群里面有朋友问博主是否改行做前端了,呵呵,其实博主是想做" ...

  9. 设计模式原则之依赖倒置原则

    所谓依赖倒置原则(Dependence Inversion Principle )就是要依赖于抽象,不要依赖于具体.简单的说就是对抽象进行编程,不要对实现进行编程,这样就降低了客户与实现模块间的耦合. ...

最新文章

  1. ubuntu下vim的配置
  2. 与Henrik Feldt谈论Suave 1.0
  3. Mysql 开启远程连接
  4. 求1e11以内的素数
  5. 《少年的你》惊现魅族手机,“心酸”回应:纯属意外 小破厂没钱植入
  6. 写在ICDsoft购买和Paypal付款之后
  7. 基于JWT(Json Web Token)的授权方式
  8. 想学习C语言,学习路线是什么?
  9. 模式识别之特征提取算法
  10. win7设置环境变量未生效
  11. 2021春招美团算法笔试题
  12. 【企业】马化腾致信合作伙伴:灰度法则的七个维度
  13. 【Python从入门到精通】(一)就简单看看Python吧
  14. Microsoft Visual C++ 6.0预处理器参考手册
  15. 未来已来 云上安全SaaS化势不可挡
  16. 非常不错的IOS学习网站
  17. 《你充满电了吗?》读后感
  18. python打印一年的日历_使用Python打印日历
  19. 创建gushi.txt文件,并写入一首古诗。
  20. 服务器无线密码是什么原因,有密码为什么连不上wifi

热门文章

  1. 安卓兼容7.0图库选择图片生成二维码
  2. 常用10个Excel快捷键,提高工作效率
  3. oracle 控制台使用手册,Oracle-ESS-入门手册
  4. 微信授权登录mock(在没有真实微信账号的情况下测试大量微信账户授权登录的情况)...
  5. 熟练的运用计算机英语怎么说,熟练的英文翻译,熟练英语怎么说
  6. ink css,CSS text-decoration-skip-ink属性用法及代码示例
  7. iOS开发 - 使用IJKPlayer时,关于需求要边下边播的缓存功能,退回来后播放缓存不再耗流量
  8. 飞书的聊天信息服务器,飞书服务端SDK java
  9. 黑鲨Android系统耗电高,已达安卓顶配,黑鲨2pro作为主力机,聊聊使用感受
  10. Percentile