软件设计原则-依赖倒置原则讲解以及代码示例

依赖倒置原则

一,介绍

1.前言

依赖倒置原则(Dependency Inversion Principle,DIP)是面向对象设计中的一个重要原则,由Robert C. Martin提出。

依赖倒置原则的核心思想是:高层模块不应该依赖于低层模块,二者都应该依赖于抽象。抽象不应该依赖于具体实现细节,而具体实现细节应该依赖于抽象。这意味着我们在进行系统设计时,应该尽量使用抽象类或接口来定义对象之间的依赖关系,而不是直接依赖于具体的实现类。

2.何时使用依赖倒置原则

  1. 在编写高层模块时:依赖倒置原则要求高层模块不应该直接依赖于低层模块,而是通过抽象接口或抽象类来进行依赖。这样可以使得高层模块与底层模块解耦,从而提高系统的灵活性和可维护性。

  2. 在进行模块间的解耦时:依赖倒置原则可以帮助我们将模块之间的依赖关系反转,从而降低了模块间的耦合度。通过引入抽象接口或抽象类作为依赖关系的中介,可以使得模块之间更加独立,易于替换和扩展。

  3. 在应用依赖注入(Dependency Injection)时:依赖注入是一种实现依赖倒置的具体方式,它通过外部将依赖对象注入到需要使用的对象中,从而减少了对象之间的直接依赖关系。依赖注入可以使得系统更加灵活,易于测试和扩展。

  4. 在进行单元测试时:依赖倒置原则可以帮助我们编写更加可测试的代码。通过将依赖对象抽象化,并使用接口或抽象类进行依赖注入,可以方便地替换依赖对象,从而使得单元测试更加简单和可靠。

二,代码示例

为了更详细地介绍依赖倒置原则,我们可以通过一个例子来说明:

假设有一个订单管理系统,系统中包含了订单类和数据库类,订单类负责处理订单相关的业务逻辑,而数据库类负责与数据库交互。最初的设计可能会像这样:
 

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. 降低模块间的耦合度:依赖倒置原则可以将模块之间的直接依赖关系转变为对抽象接口或抽象类的依赖,从而降低了模块之间的耦合度。这样,当一个模块发生变化时,对其他模块的影响也会减少,提高了系统的灵活性。

  2. 提高代码的可扩展性:通过引入抽象接口或抽象类,依赖倒置原则使得系统更易于扩展。当需要添加新的功能时,只需要针对抽象进行扩展,而不需要修改现有的代码。这有助于降低对原有代码的影响范围,提高了代码的可维护性和可复用性。

  3. 促进代码的测试和调试:依赖倒置原则可以帮助我们编写更加可测试和可调试的代码。通过引入抽象接口或抽象类,并使用依赖注入等机制,可以方便地替换依赖对象,从而使得单元测试更加简单和可靠。

  4. 支持多态性:依赖倒置原则支持多态性的应用。通过面向抽象编程,可以针对抽象类型进行编程,而不关心具体的实现类。这样可以提高代码的灵活性和可复用性。

缺点:

  1. 增加了系统的复杂性:引入抽象接口或抽象类,以及依赖注入等机制,会增加系统的复杂性。需要额外的设计和开发工作来定义和管理抽象接口,并实现依赖注入。

  2. 需要对系统进行全面设计:依赖倒置原则需要在系统设计的早期考虑,需要合理划分抽象接口和实现类,并建立合适的依赖关系。如果在系统设计已经完成的情况下才引入依赖倒置原则,可能需要大量的重构工作。

  3. 可能增加运行时的性能开销:由于依赖倒置原则需要通过抽象接口进行运行时的依赖解析和注入,可能会带来一定的性能开销。特别是在系统规模较大且依赖关系较复杂的情况下,可能需要额外的开销来管理依赖关系。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/115632.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

aiohttp ssl.SSLError: [SSL: SSLV3_ALERT_HANDSHAKE_FAILURE] 错误处理

这个问题原因吧其实就是3.10 开始官网更新了TLS 堆栈默认安全设置 感兴趣的可以看下链接 python官网叙述: Python 3.10 增加了 TLS 堆栈的默认安全设置 解决也很简单,将ssl安全等级降下来就行,例如: import ssl import aiohttp ctx ssl.cr…

Redis 底层对 String 的 3 个优化

Redis对 String 类型实现了很多优化,通过以下三个重要的优化点来解释: 1. 简单动态字符串(SDS) Redis 的 String 类型内部采用简单动态字符串(SDS)来管理字符串。相比于 C 语言的原生字符串,S…

二、【MyBatis】 MyBatis入门与简单使用

二、【MyBatis】 MyBatis入门与简单使用 二、【MyBatis】 MyBatis入门与简单使用一、什么是ORM二、为什么mybatis是半自动的ORM框架2.1 Hibernate优点2.2 Hibernate缺点2.3 MyBatis与Hibernate区别三、Mybatis快速入门3.1 项目引入Maven相关依赖3.2 创建测试数据库3.3 编写数据…

sealos一键部署K8S环境(sealos3.0时代教程过时了,目前已经4.0了,请移步使用Sealos一键安装K8S)

1 安装Sealos(4.0版本) sealos部署k8s贼方便,只需要一条init命令即可,3分钟部署完(下载安装包的时间不算)。 官方教程:https://www.sealyun.com/instructions/1st #主机名: hostnamectl set-hostname mas…

PROSTATEx-2 上前列腺癌的 3D CNN 分类

内容 本文介绍了在多参数 MRI 序列上使用 3D CNN 对前列腺癌进行显着性或不显着性分类。内容如下: 数据集描述Dicom 到 Nifti 文件格式的转换不同 MRI 序列的联合配准

redis底层数据结构

总所周知,redis支持五种数据类型String、Hash、List、Set、ZSet。在支持这些复杂数据结构的同时,redis不仅需要保证读写的性能,还能提供各种微操作,比如直接修改Hash字典中的某个field的值,或者直接往ZSet中插入某个值…

【人工智能Ⅰ】2-知识表示

【人工智能Ⅰ】2-知识表示 知识是智能的基础 一阶谓词逻辑、产生式、框架等知识表示方法 文章目录 【人工智能Ⅰ】2-知识表示2.1 知识与知识表示的概念知识的概念知识的特性 2.2 一阶谓词逻辑表示法谓词谓词公式一阶谓词逻辑知识表示方法一阶谓词逻辑表示法的特点 2.3 产生式…

Failed to start The nginx HTTP and reverse proxy server.

本章教程主要分享一下,当nginx 启动时,遇到报这个错误时的一个解决问题思路。 目录 1、观察报错信息 2、尝试性解决 1、观察报错信息 根据日志的信息,我们至少可以知道2个比较信息。 1、操作用户执行命令是在非root权限下进行操作的。 2、Ad…

Xcode14创建github远程仓库Token

1.点击Create a Token on GitHub 2.在打开的网页中,登陆GitHub 3.点击生成Token 这是不能为空 4.Token创建成功如下: 5.复制Token到Xcode然后点击Sign In登陆 正在创建远程我仓库 正在将本地仓库代码推入远程仓库 创建成功

C++项目——云备份-②-第三方库认识

文章目录 专栏导读1. json 认识1.1 JSON 数据结构的特点 2. jsoncpp库认识3. json实现序列化案例4. json实现反序列化案例5. bundle文件压缩库认识6. bundle库实现文件压缩案例7.bundle库实现文件解压缩案例8.httplib库认识9. httplib库搭建简单服务器案例10. httplib库搭建简单…

YOLO目标检测——密集人群人头检测数据集【含对应voc、coco和yolo三种格式标签】

实际项目应用:在公共场所,如车站、商场、景区等,可以通过人头目标检测技术来监测人群流量数据集说明:人头检测数据集,真实场景的高质量图片数据,数据场景丰富标签说明:使用lableimg标注软件标注…

聊聊KafkaListener的实现机制

序 本文只要研究一下KafkaListener的实现机制 KafkaListener org/springframework/kafka/annotation/KafkaListener.java Target({ ElementType.TYPE, ElementType.METHOD, ElementType.ANNOTATION_TYPE }) Retention(RetentionPolicy.RUNTIME) MessageMapping Documented …

python实现excel的数据提取

一文带你实现excel表格的数据提取 今天记录一下如何使用python提取Excel中符合特定条件的数据 在数据处理和分析的过程中,我们经常需要从Excel表格中提取特定条件下的数据。Python的pandas库为我们提供了方便的方法来进行数据查询和过滤。 Pandas 是 Python 语言…

SELECT COUNT(*) 会造成全表扫描吗?

前言 SELECT COUNT(*)会不会导致全表扫描引起慢查询呢? SELECT COUNT(*) FROM SomeTable 网上有一种说法,针对无 where_clause 的 COUNT(*),MySQL 是有优化的,优化器会选择成本最小的辅助索引查询计数,其实反而性能…

LeetCode 面试题 10.10. 数字流的秩

文章目录 一、题目二、C# 题解 一、题目 假设你正在读取一串整数。每隔一段时间,你希望能找出数字 x 的秩(小于或等于 x 的值的个数)。请实现数据结构和算法来支持这些操作,也就是说: 实现 track(int x) 方法,每读入一个数字都会调…

基于深度学习实现一张单图,一个视频,一键换脸,Colab脚本使用方法,在线版本,普通人也可以上传一张图片体验机器学习一键换脸

基于深度学习实现一张单图,一个视频,一键换脸,Colab脚本使用方法,在线版本,普通人也可以上传一张图片体验机器学习一键换脸。 AI领域人才辈出,突然就跳出一个大佬“s0md3v”,开源了一个单图就可以进行视频换脸的项目。 项目主页给了一张换脸动图非常有说服力,真是一图…

aarch64-poky-linux-gcc交叉编译报错缺少头文件.h解决方案

0x00 前言 演示用操作系统:Ubuntu 18.04.6 LTS \n \l aarch64-poky-linux-gcc版本:gcc version 10.2.0 (GCC) 最后更新日期:2023-10-19 0x01 解决方案 aarch64-poky-linux-gcc交叉编译时报错缺少头文件.h可能是因为没有设置--sysroot编译…

JavaScript从入门到精通系列第二十三篇:JavaScript中的数组

前言 1:对象的分类 在JavaScript中,对象可以分为以下几种类型: 内置对象:这些对象是由JavaScript引擎提供的,如Object、Array、String、Date、RegExp等。 宿主对象:这些对象是由宿主环境(浏览…

Fiber Golang:Golang中的强大Web框架

揭示Fiber在Go Web开发中的特点和优势 在不断发展的Web开发领域中,选择正确的框架可以极大地影响项目的效率和成功。介绍一下Fiber,这是一款令人印象深刻的Golang(Go语言)Web框架。以其飞快的性能和强大的特性而闻名,…

【算法训练-动态规划 一】【应用DP问题】零钱兑换、爬楼梯、买卖股票的最佳时机I、打家劫舍

废话不多说,喊一句号子鼓励自己:程序员永不失业,程序员走向架构!本篇Blog的主题是【动态规划】,使用【数组】这个基本的数据结构来实现,这个高频题的站点是:CodeTop,筛选条件为&…