依赖倒置原则(Dependence Inversion Priiciple,DIP)

介绍

High level modules should not depend upon low level modules. Both should depend upon abstractions. Abstractions should not depend upon details. Detatils should depend upon abstractions.

翻译过来有是三个意思:

1.高层模块不应该依赖底层模块,两者都应该依赖其抽象。

2.抽象不应该依赖细节。

3.细节应该依赖抽象。

在Java中,表现就是:

1.模块间的依赖通过抽象发生,实现类之间不应该发生直接的依赖关机,其依赖关系应该是用过接口或者抽象类产生的;

2.接口或者抽象类不应该依赖实现类;

3.实现类应该依赖接口或者抽象类。

依赖倒置原则的设计理念是:相较于实习类具有多变性,抽象要稳定的多。框架的设计应该基于抽象,这样子,能提高项目的稳定性。

在框架中,抽象的作用是用来制定规范,具体的细节应该交给实现类。

应用实例

有一个司机类对象张三(zs),具有drive方法来驾驶汽车。

public class Dependence1 {
public static void main(String[] args) {
Driver zs = new Driver();
Benz benz = new Benz();
zs.drive(benz);
}
} class Driver {
public void drive(Benz benz) {
benz.run();
}
} class Benz {
public void run() {
System.out.println("奔驰汽车发动...");
}
} //经过艰苦奋斗,张三同志买了一辆宝马汽车
class BMW {
public void run() {
System.out.println("宝马汽车发动...");
}
}

目前为止,看上去很美好,正常的功能也实现了;但是,因为这个功能是基于具体的实现类Benz实现的,这就为后来的扩展带来了麻烦:比如张三同志艰苦奋斗,又买了一辆宝马(BMW),但是根据Driver类的drive方法,张三同志不能驾驶新买的宝马汽车。这就是很严重的逻辑了:只要是有了驾照,奔驰和宝马应该都能驾驶。

改进措施:

很明显,司机类中的drive方法不应该依赖于一种具体品牌的汽车,而因该是汽车的泛指。所以,新建两个接口:

IDriver和ICar,分别定义了司机和汽车的职能,汽车就是驾驶汽车。具体的代码如下:

public class Dependence2 {
public static void main(String[] args) {
IDriver zs = new Driver();
ICar benz = new Benz();
zs.drive(benz);
ICar bmw = new BMW();
zs.drive(bmw);
}
} interface ICar {
//是汽车都应该具备的功能
public void run();
} class Benz implements ICar {
@Override
public void run() {
System.out.println("奔驰汽车发动...");
}
} class BMW implements ICar {
@Override
public void run() {
System.out.println("宝马汽车发动...");
}
} interface IDriver {
//是司机就应该会开车,不管是什么车
public void drive(ICar iCar);
} class Driver implements IDriver {
@Override
public void drive(ICar iCar) {
iCar.run();
}
}

在demo中,类Dependence2 作为高层模块,它对底层模块的依赖建立在抽象上;司机张三的表面类型时IDriver,Benz和BMW的表面类型是ICar,这样做,在扩展业务逻辑(比如新增一辆汽车Mini)时,只需修改业务场景类(高层模块),对底层模块如Driver和Benz等没有影响。

依赖的三种形式

依赖的形式有三种:接口依赖、setter方法依赖和构造器依赖。具体介绍如下:

1.接口依赖

顾名思义,就是通过实现接口的方式注入依赖,代码详见上面的改进措施。

2.setter方法依赖

在抽象中定义setter方法,就是所谓的setter方法依赖。

demo如下:

interface IDriver {
//setter方式注入
public void setICar(ICar iCar); } class Driver implements IDriver {
private ICar iCar; @Override
public void setICar(ICar iCar) {
this.iCar = iCar;
} }

3.构造器依赖

这个也比较好理解,就是将ICar的具体类型,通过构造器传给Driver:

class Driver implements IDriver {
private ICar iCar; public Driver(ICar iCar) {
this.iCar =iCar;
}
}

注意事项和细节

在项目中,底层模块的类都应该有抽象类或者接口,或者两者都有;

变量的声明应该是抽象类型,有利于提高项目的稳定性;

任何类都不应该从具体类派生;

尽量不要重写积累的方法,要结合“里氏替换原则”使用。

DesignPattern系列__03依赖倒置原则的更多相关文章

  1. 深入理解JavaScript系列(22):S.O.L.I.D五大原则之依赖倒置原则DIP

    前言 本章我们要讲解的是S.O.L.I.D五大原则JavaScript语言实现的第5篇,依赖倒置原则LSP(The Dependency Inversion Principle ). 英文原文:htt ...

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

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

  3. 依赖倒置原则(Dependency Inversion Principle)

    很多软件工程师都多少在处理 "Bad Design"时有一些痛苦的经历.如果发现这些 "Bad Design" 的始作俑者就是我们自己时,那感觉就更糟糕了.那么 ...

  4. 对依赖倒置原则(DIP)及Ioc、DI、Ioc容器的一些理解

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

  5. 第2章 面向对象的设计原则(SOLID):3_依赖倒置原则(DIP)

    3. 依赖倒置原则(Dependence Inversion Principle,DIP) 3.1 定义 (1)要依赖抽象,不要依赖具体的实现类.简单的说就是对抽象(或接口)进行编程,不要依赖实现进行 ...

  6. 3.依赖倒置原则(Dependence Inversion Principle)

    1.定义 高层模块不应该依赖于低层模块,二者都应该依赖于抽象:抽象不应该依赖细节:细节应该依赖抽象. 2.定义解读 依赖倒置原则在程序编码中经常运用,其核心思想就是面向接口编程,高层模块不应该依赖低层 ...

  7. 【设计模式六大原则3】依赖倒置原则(Dependence Inversion Principle)

      定义:高层模块不应该依赖低层模块,二者都应该依赖其抽象:抽象不应该依赖细节:细节应该依赖抽象. 问题由来:类A直接依赖类B,假如要将类A改为依赖类C,则必须通过修改类A的代码来达成.这种场景下,类 ...

  8. [设计模式]<<设计模式之禅>>关于依赖倒置原则

    依赖倒置原则(Dependence Inversion Principle,DIP)这个名字看着有点别扭,“依赖”还“倒置”,这到底是什么意思?依赖倒置原则的原始定义是 High level modu ...

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

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

随机推荐

  1. TypeScript算法与数据结构-数组篇

    数组是数据结构中最简单,也是使用最广泛的一种.在原生的js中,数组给我们提供了很多方便的操作方法,比如push(), pop(), shift(), unshift().但是出于对数据结构的学习,我们 ...

  2. SQLServer性能优化之---数据库级日记监控

    上节回顾:https://www.cnblogs.com/dotnetcrazy/p/11029323.html 4.6.6.SQLServer监控 脚本示意:https://github.com/l ...

  3. spark入门(二)RDD基础操作

    1 简述 spark中的RDD是一个分布式的元素集合. 在spark中,对数据的所有操作不外乎创建RDD,转化RDD以及调用RDD操作进行求值,而这些操作,spark会自动将RDD中的数据分发到集群上 ...

  4. Codeforces Gym100502A:Amanda Lounges(DFS染色)

    http://codeforces.com/gym/100502/attachments 题意:有n个地点,m条边,每条边有一个边权,0代表两个顶点都染成白色,2代表两个顶点都染成黑色,1代表两个顶点 ...

  5. Java连接MYSQL进行操作(增,删,改)

    连接数据库,并用表格输出数据 创建insex.jsp <table border="1" width="80%" align='center'> & ...

  6. Java 中的字符串(String)与C# 中字符串(string)的异同

    1. C# 中比较两个字符串字面量是否相等,可以使用 “==”比较运算符,是因为string 类型重写(override)了“==” 和 “!=” 运算符,在使用“==” 和 “!=” 进行字符串比较 ...

  7. 01(b)无约束优化(准备知识)

    1.解方程转化为优化问题 $n\left\{ \begin{aligned}& {{P}_{1}}(x)=0 \\ & {{P}_{2}}(x)=0 \\ & \text{   ...

  8. 按行读取String类型

    BufferedReader br = new BufferedReader(new InputStreamReader(new ByteArrayInputStream(content.getByt ...

  9. Java基础之增强for循环

    平时大家for循环应该用的不少,特别是增强for循环,简单快捷.但是在增强for中做删除操作,却会抛出java.util.ConcurrentModificationException,一起来看下. ...

  10. GPS常识-A版(详)

    第一章 绪论 1.简述GPS系统的特点有哪些? GPS在测绘工程中应用的优点 P13 ●定位精度高 应用实践证明,相对静态定位1小时以上观测解,其平面位置:在300-1500m范围内,绝对误差小于1m ...