DesignPattern系列__03依赖倒置原则
依赖倒置原则(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依赖倒置原则的更多相关文章
- 深入理解JavaScript系列(22):S.O.L.I.D五大原则之依赖倒置原则DIP
前言 本章我们要讲解的是S.O.L.I.D五大原则JavaScript语言实现的第5篇,依赖倒置原则LSP(The Dependency Inversion Principle ). 英文原文:htt ...
- C#软件设计——小话设计模式原则之:依赖倒置原则DIP
前言:很久之前就想动笔总结下关于软件设计的一些原则,或者说是设计模式的一些原则,奈何被各种bootstrap组件所吸引,一直抽不开身.群里面有朋友问博主是否改行做前端了,呵呵,其实博主是想做“全战”, ...
- 依赖倒置原则(Dependency Inversion Principle)
很多软件工程师都多少在处理 "Bad Design"时有一些痛苦的经历.如果发现这些 "Bad Design" 的始作俑者就是我们自己时,那感觉就更糟糕了.那么 ...
- 对依赖倒置原则(DIP)及Ioc、DI、Ioc容器的一些理解
1.概述 所谓依赖倒置原则(Dependence Inversion Principle)就是要依赖于抽象,不要依赖于具体.简单的说就是要求对抽象进行编程,不要对实现进行编程,这样就降低了客户与实现模 ...
- 第2章 面向对象的设计原则(SOLID):3_依赖倒置原则(DIP)
3. 依赖倒置原则(Dependence Inversion Principle,DIP) 3.1 定义 (1)要依赖抽象,不要依赖具体的实现类.简单的说就是对抽象(或接口)进行编程,不要依赖实现进行 ...
- 3.依赖倒置原则(Dependence Inversion Principle)
1.定义 高层模块不应该依赖于低层模块,二者都应该依赖于抽象:抽象不应该依赖细节:细节应该依赖抽象. 2.定义解读 依赖倒置原则在程序编码中经常运用,其核心思想就是面向接口编程,高层模块不应该依赖低层 ...
- 【设计模式六大原则3】依赖倒置原则(Dependence Inversion Principle)
定义:高层模块不应该依赖低层模块,二者都应该依赖其抽象:抽象不应该依赖细节:细节应该依赖抽象. 问题由来:类A直接依赖类B,假如要将类A改为依赖类C,则必须通过修改类A的代码来达成.这种场景下,类 ...
- [设计模式]<<设计模式之禅>>关于依赖倒置原则
依赖倒置原则(Dependence Inversion Principle,DIP)这个名字看着有点别扭,“依赖”还“倒置”,这到底是什么意思?依赖倒置原则的原始定义是 High level modu ...
- ASP.NET 设计模式中依赖倒置原则
依赖倒置原则 A.高层次的模块不应该依赖于低层次的模块,他们都应该依赖于抽象. B.抽象不应该依赖于具体,具体应该依赖于抽象. 依赖倒置原则 A.高层次的模块不应该依赖于低层次的模块,他们都应该依赖于 ...
随机推荐
- vue接入萤石云视频
在萤石云开放平台注册开发者账号,网址:https://open.ys7.com/guide.html 在“开发者服务”-->“我的设备”里添加已有设备或者申请一个试用设备 然后点击“我的应用”里 ...
- 解决 Prometheus 不能获取 Kubernetes 集群上 Windows 节点的 Metrics
背景 接上一篇 快速搭建 Windows Kubernetes , 我们发现原来在 Windows Kubernetes 会有一些与在 Linux 上使用不一样的体验,俗称坑,例如 hostAlias ...
- easyui datagrid 单元格 编辑时 事件 修改另一单元格
//datagrid 列数据 $('#acc').datagrid({ columns : [ [ { field : 'fee_lend', title : '收费A', width : 100, ...
- 从无到有构建vue实战项目(四)
六.webpack的安装和配置 为了快速构建vue项目,webpack是一个必不可少的工具,我们先来安装它,附上官网地址:https://www.webpackjs.com/ 要安装最新版本或特定版本 ...
- Easy doesn't enter into grown-up life
Hi xinlu, good afternoon, my name is xx, you can also call me andrew. i'm 31 years old, and i'm from ...
- leetcode 136 Single Number bBt Option
Linked Url:https://leetcode.com/problems/single-number/ Given a non-empty array of integers, every e ...
- 数据结构丨N叉树
遍历 N叉树的遍历 树的遍历 一棵二叉树可以按照前序.中序.后序或者层序来进行遍历.在这些遍历方法中,前序遍历.后序遍历和层序遍历同样可以运用到N叉树中. 回顾 - 二叉树的遍历 前序遍历 - 首先访 ...
- linux字符设备驱动--基本知识介绍
一.设备驱动的分类 1.字符设备 字符设备是指那些能一个字节一个字节读取数据的设备,如LED灯.键盘.鼠标等.字符设备一般需要在驱动层实现open().close().read().write().i ...
- 【最短路算法例题-升降梯上】-C++
描述 启了升降梯的动力之后,探险队员们进入了升降梯运行的那条竖直的隧道,映入眼帘的是一条直通塔顶的轨道.一辆停在轨道底部的电梯.和电梯内一杆控制电梯升降的巨大手柄. Nescafe之塔一共有N层,升降 ...
- Makefile简单的配置
一.文件配置目录 1)原目录 demo ├── Makefile ├── demo.cpp ├── demo.hpp └── main.cpp 2)make之后的目录 demo ├── Makefil ...