软件设计原则-依赖倒置原则讲解以及代码示例
依赖倒置原则
一,介绍
1.前言
依赖倒置原则(Dependency Inversion Principle,DIP)是面向对象设计中的一个重要原则,由Robert C. Martin提出。
依赖倒置原则的核心思想是:高层模块不应该依赖于低层模块,二者都应该依赖于抽象。抽象不应该依赖于具体实现细节,而具体实现细节应该依赖于抽象。这意味着我们在进行系统设计时,应该尽量使用抽象类或接口来定义对象之间的依赖关系,而不是直接依赖于具体的实现类。
2.何时使用依赖倒置原则
-
在编写高层模块时:依赖倒置原则要求高层模块不应该直接依赖于低层模块,而是通过抽象接口或抽象类来进行依赖。这样可以使得高层模块与底层模块解耦,从而提高系统的灵活性和可维护性。
-
在进行模块间的解耦时:依赖倒置原则可以帮助我们将模块之间的依赖关系反转,从而降低了模块间的耦合度。通过引入抽象接口或抽象类作为依赖关系的中介,可以使得模块之间更加独立,易于替换和扩展。
-
在应用依赖注入(Dependency Injection)时:依赖注入是一种实现依赖倒置的具体方式,它通过外部将依赖对象注入到需要使用的对象中,从而减少了对象之间的直接依赖关系。依赖注入可以使得系统更加灵活,易于测试和扩展。
-
在进行单元测试时:依赖倒置原则可以帮助我们编写更加可测试的代码。通过将依赖对象抽象化,并使用接口或抽象类进行依赖注入,可以方便地替换依赖对象,从而使得单元测试更加简单和可靠。
二,代码示例
为了更详细地介绍依赖倒置原则,我们可以通过一个例子来说明:
假设有一个订单管理系统,系统中包含了订单类和数据库类,订单类负责处理订单相关的业务逻辑,而数据库类负责与数据库交互。最初的设计可能会像这样:
class Order {private Database database;// 省略构造方法和其他属性方法public void save() {database.save(this);}
}class Database {public void save(Order order) {// 保存订单到数据库}
}
在这个设计中,订单类直接依赖于具体的数据库类,这样一来,如果将来需要更换数据库操作方式,就需要修改订单类的代码,违反了开闭原则。
为了符合依赖倒置原则,我们可以进行重构。首先,定义一个抽象类或接口`Database`:
interface Database {void save(Order order);
}
然后,订单类通过构造函数或setter方法注入一个`Database`对象,从而将具体的数据库实现与订单类解耦:
class Order {private Database database;public Order(Database database) {this.database = database;}// 省略其他属性方法public void save() {database.save(this);}
}
当需要使用特定的数据库实现时,我们只需要创建一个实现了`Database`接口的具体类,并将其传递给订单类:
这样,订单类与具体的数据库实现解耦,而且我们可以轻松地使用不同的数据库实现,而无需修改订单类的代码。
依赖倒置原则的目的是降低模块之间的耦合度,提高系统的灵活性和可维护性。通过面向抽象编程,并通过依赖注入的方式实现对象之间的解耦,可以使系统更容易扩展和修改,同时也方便进行单元测试和模块替换。
总结起来,依赖倒置原则要求我们将高层模块的设计依赖于抽象,而不是具体实现细节。它是面向对象设计中的重要原则之一,在软件开发中具有广泛的适用性和重要性。
三,优缺点
优点:
-
降低模块间的耦合度:依赖倒置原则可以将模块之间的直接依赖关系转变为对抽象接口或抽象类的依赖,从而降低了模块之间的耦合度。这样,当一个模块发生变化时,对其他模块的影响也会减少,提高了系统的灵活性。
-
提高代码的可扩展性:通过引入抽象接口或抽象类,依赖倒置原则使得系统更易于扩展。当需要添加新的功能时,只需要针对抽象进行扩展,而不需要修改现有的代码。这有助于降低对原有代码的影响范围,提高了代码的可维护性和可复用性。
-
促进代码的测试和调试:依赖倒置原则可以帮助我们编写更加可测试和可调试的代码。通过引入抽象接口或抽象类,并使用依赖注入等机制,可以方便地替换依赖对象,从而使得单元测试更加简单和可靠。
-
支持多态性:依赖倒置原则支持多态性的应用。通过面向抽象编程,可以针对抽象类型进行编程,而不关心具体的实现类。这样可以提高代码的灵活性和可复用性。
缺点:
-
增加了系统的复杂性:引入抽象接口或抽象类,以及依赖注入等机制,会增加系统的复杂性。需要额外的设计和开发工作来定义和管理抽象接口,并实现依赖注入。
-
需要对系统进行全面设计:依赖倒置原则需要在系统设计的早期考虑,需要合理划分抽象接口和实现类,并建立合适的依赖关系。如果在系统设计已经完成的情况下才引入依赖倒置原则,可能需要大量的重构工作。
-
可能增加运行时的性能开销:由于依赖倒置原则需要通过抽象接口进行运行时的依赖解析和注入,可能会带来一定的性能开销。特别是在系统规模较大且依赖关系较复杂的情况下,可能需要额外的开销来管理依赖关系。
相关文章:

软件设计原则-依赖倒置原则讲解以及代码示例
依赖倒置原则 一,介绍 1.前言 依赖倒置原则(Dependency Inversion Principle,DIP)是面向对象设计中的一个重要原则,由Robert C. Martin提出。 依赖倒置原则的核心思想是:高层模块不应该依赖于低层模块&…...

Linux--进程替换
1.什么是进程替换 在fork函数之后,父子进程各自执行代码的一部分,但是如果子进程想要执行一份全新的程序呢? 通过进程替换来完成,进程替换就是父子进程代码发生写时拷贝,子进程执行自己的功能。 程序替换就是通过特定的…...

【计算机网络】TCP协议
文章目录 1. TCP报文的结构2. TCP的发送缓冲区和接收缓冲区3. 确保可靠性序列号和确认序列号确认应答超时重传连接管理1️⃣三次握手建立连接2️⃣四次挥手断开连接 4. 提高性能流量控制滑动窗口拥塞控制延迟应答捎带应答 5. 面向字节流6. TCP/UDP对比 概念:TCP&…...

机器学习数据集:Kaggle
什么是Kaggle? Kaggle成立于2010年,是一个进行数据发掘和预测竞赛的在线平台。从公司的角度来讲,可以提供一些数据,进而提出一个实际需要解决的问题;从参赛者的角度来讲,他们将组队参与项目,针…...

软考 系统架构设计师系列知识点之设计模式(4)
接前一篇文章:软考 系统架构设计师系列知识点之设计模式(3) 所属章节: 老版(第一版)教材 第7章. 设计模式 第2节. 设计模式实例 3. 行为型模式 行为型模式可以影响一个系统的状态和行为流。通过优化状态…...

PyCharm 安装 cx_Oracle 失败
我在PyCharm的终端用 pip安装cx_Oracle失败,报错情况如下: ERROR: Could not build wheels for cx_Oracle, which is required to install pyproject.toml-based projects 出错原因: python 的版本太高了,我的是3.11版本的&…...

解决Windows出现找不到mfcm90u.dll无法打开软件程序的方法
今天,我非常荣幸能够在这里与大家分享关于mfc90u.dll丢失的5种解决方法。在我们日常使用电脑的过程中,可能会遇到一些软件或系统错误,其中之一就是mfc90u.dll丢失。那么,mfc90u.dll究竟是什么文件呢?接下来,…...

如何设计线程安全的 HashMap?
如何设计线程安全的 HashMap? HashMap 线程不安全的体现: 多线程下扩容死循环:JDK1.7中的 HashMap 使用头插法插入元素,在多线程的环境下,扩容的时候有可能导致环形链表的出现,形成死循环。因此,JDK1.8使…...

rpc汇总
1、什么是rpc rpc的应用,有哪些 Google 开源了 gRPC, Facebook 开源了 Thrift, Twitter 开源了 Finagle, 百度开源了bRPC, 腾讯开源了 Tars, 阿里开源了 Dubbo 和 HSF, 新浪开源了 Motan 等 gr…...

OpenCV学习(五)——图像基本操作(访问图像像素值、图像属性、感兴趣区域ROI和图像边框)
图像基本操作 5. 图像基本操作5.1 访问像素值并修改5.2 访问图像属性5.2 图像感兴趣区域ROI5.3 拆分和合并图像通道5.4 为图像设置边框(填充) 5. 图像基本操作 访问像素值并修改访问图像属性设置感兴趣区域(ROI)分割和合并图像 …...

指针仪表读数YOLOV8NANO
指针仪表读数YOLOV8 NANO 采用YOLOV8 NANO训练,标记,然后判断角度,得出角度,可以通过角度,换算成数据...

10000字!图解机器学习特征工程
文章目录 引言特征工程1.特征类型1.1 结构化 vs 非结构化数据1.2 定量 vs 定性数据 2.数据清洗2.1 数据对齐2.2 缺失值处理 原文链接:https://www.showmeai.tech/article-detail/208 作者:showmeAI 引言 上图为大家熟悉的机器学习建模流程图,…...

Java 官方提供了哪几种线程池,分别有什么特点?
JDK 中提供了 5 中不同线程池的创建方式: newCachedThreadPool newCachedThreadPool, 是一种可以缓存的线程池,它可以用来处理大量短期的突发流量。 它的特点有三个,最大线程数是 Integer.MaxValue,线程存活时间是 60 …...

DTI-ALPS处理笔记
DTI-ALPS处理笔记 前言: 前段时间刚好学习了一下DTI-ALPS处理(diffusion tensor image analysis along the perivascular space ),记录一下,以便后续学习。ALPS是2017年发表在《Japanese Journal of Radiology》的一篇文章首次提出的 (文章地址),主要用于无创评估脑内淋…...

LVS集群-NAT模式
集群的概念: 集群:nginx四层和七层动静分离 集群标准意义上的概念:为解决特定问题将多个计算机组合起来形成一个单系统 集群的目的就是为了解决系统的性能瓶颈。 垂直扩展:向上扩展,增加单个机器的性能,…...

微服务技术导学
文章目录 微服务结构认识微服务技术栈 微服务结构 技术: 解决异常定位: 持续集成,解决自动化的部署: 总结如下: 认识微服务 微服务演变: 技术栈 SpringCloud与SpringBoot版本对应关系...

p5.js 开发点彩画派的绘画工具
本文简介 点赞 关注 收藏 学会了 这几天在整理书柜时看到这套书,看到梵高,想起他的点彩画。 想到点彩画派,不得不提的一个画家叫乔治皮埃尔秀拉。据说梵高也模仿过他的画作。 我引用一下维基百科对点彩画派的解析: 点彩画派&…...

Java工具库——Commons IO的50个常用方法
工具库介绍 Commons IO(Apache Commons IO)是一个广泛用于 Java 开发的开源工具库,由Apache软件基金会维护和支持。这个库旨在简化文件和流操作,提供了各种实用工具类和方法,以便更轻松地进行输入输出操作。以下是 Com…...

Git: 仓库clone和用户配置
git clone 两种方式clone远程仓库到本地。 通过ssh 命令格式: git clone gitxxxxxx.git使用这种方法需要提前创建ssh秘钥。 首先打开一个git控制台,输入命令 ssh-keygen -t ed25519 -C “xxxxxxxxxx.com”输入命令后需要点击四次回车,其…...

构建外卖小程序:技术要点和实际代码
1. 前端开发 前端开发涉及用户界面设计和用户交互。HTML、CSS 和 JavaScript 是构建外卖小程序界面的主要技术。 <!-- HTML 结构示例 --> <header><h1>外卖小程序</h1><!-- 其他导航元素 --> </header> <main><!-- 菜单显示 -…...

ubuntu安装配置svn
目录 简介安装SVN 启动模式方式1:单库svnserve方式方式2:多库svnserve方式 SVN 创建版本库1.svn 服务配置文件 svnserve.conf2.用户名口令文件 passwd3.权限配置文件4.多库方式运行 SVN 检出操作SVN 解决冲突SVN 提交操作SVN 版本回退SVN 查看历史信息1.svn log2.svn diff3.svn…...

『Jmeter入门万字长文』 | 从环境搭建、脚本设计、执行步骤到生成监控报告完整过程
『Jmeter入门万字长文』 | 从环境搭建、脚本设计、执行步骤到生成监控报告完整过程 1 Jmeter安装1.1 下载安装1.2 Jmeter汉化1.2.1 临时修改1.2.2 永久修改 1.3 验证环境 2 测试对象2.1 测试对象说明2.2 测试对象安装2.2.1 下载安装2.2.2 启动测试对象服务2.2.3 访问测试对象2.…...

Unity C#中LuaTable、LuaArrayTable、LuaDictTable中数据的增删改查
LuaTable、LuaArrayTable、LuaDictTable中数据的增删改查 介绍Lua表lua表初始化lua移除引用lua中向表中添加数据lua中表中移除数据lua表中连接数据lua表中数据排序获取lua表长度获取表中最大值 UnityC#中LuaTableUnityC#中LuaArrayTable、LuaDictTable、LuaDictTable<K,V>…...

Spring常见面试题
https://blog.csdn.net/a745233700/article/details/80959716?ops_request_misc%257B%2522request%255Fid%2522%253A%2522169847982516800213061720%2522%252C%2522scm%2522%253A%252220140713.130102334…%2522%257D&request_id169847982516800213061720&biz_id0&…...

通过Vue自带服务器实现Ajax请求跨域(vue-cli)
通过Vue自带服务器实现Ajax请求跨域(vue-cli) 跨域 原理:从A页面访问到B页面,并且要获取到B页面上的数据,而两个页面所在的端口、协议和域名中哪怕有一个不对等,那么这种行为就叫跨域。注意:类…...

Vue2-计算属性的用法
题记 vue2计算属性的用法 反转字符串 <!DOCTYPE html> <html> <head> <meta charset"utf-8"> <title>实例</title> <script src"https://cdn.staticfile.org/vue/2.2.2/vue.min.js"></script> </hea…...

SM3加密udf
SM3加密udf maven xml <dependencies> <!-- 配置日志 --> <dependency> <groupId>org.slf4j</groupId> <artifactId>slf4j-api</artifactId> <version>1.7.5</version> </dep…...

ce从初阶到大牛(两台主机免密登录)
一、配置ssh远程连接 实现两台linux主机之间通过公钥验证能够互相实现免密登陆 1.确认服务程序是否安装 rpm -qa | grep ssh 2.是否启动 ps -aux | grep ssh 3.生成非对称公钥 ssh-keygen -t rsa 4.公钥发送到客户端 cd /root/.ssh/ ssh-copy-id root192.168.170.134 因为…...

CS224W2.3——传统基于特征的方法(图层级特征)
前两篇中我们讨论了节点层级的特征表示、边层级的特征表示: CS224W2.1——传统基于特征的方法(节点层级特征)CS224W2.2——传统基于特征的方法(边层级特征) 在这篇中,我们将重点从整个图中提取特征。换句话说,我们想要描述整个图结构的特征…...

【CSS】包含块
CSS规范中的包含块 包含块的内容: 元素的尺寸和位置,会受它的包含块所影响。 对于一些属性,例如 width, height, padding, margin,绝对定位元素的偏移值(比如 position 被设置为 absolute 或 fixed)&…...