【Python】新手入门学习:详细介绍依赖倒置原则(DIP)及其作用、代码示例
【Python】新手入门学习:详细介绍依赖倒置原则(DIP)及其作用、代码示例

🌈 个人主页:高斯小哥
🔥 高质量专栏:Matplotlib之旅:零基础精通数据可视化、Python基础【高质量合集】、PyTorch零基础入门教程👈 希望得到您的订阅和支持~
💡 创作高质量博文(平均质量分92+),分享更多关于深度学习、PyTorch、Python领域的优质内容!(希望得到您的关注~)
| 博客链接 | 简要说明 |
|---|---|
| 【Python】新手入门学习:详细介绍单一职责原则(SRP)及其作用、代码示例 | 一个类应该只有一个引起变化的原因,确保类的职责单一。 |
| 【Python】新手入门学习:详细介绍开放封闭原则(OCP)及其作用、代码示例 | 软件实体应对扩展开放,对修改封闭,提高系统的可维护性和可扩展性。 |
| 【Python】新手入门学习:详细介绍里氏替换原则(LSP)及其作用、代码示例 | 子类必须能够替换其父类,且替换后,程序的行为没有变化。 |
| 【Python】新手入门学习:详细介绍依赖倒置原则(DIP)及其作用、代码示例 | 高层模块不应该依赖于低层模块,二者都应该依赖于抽象。 |
| 【Python】新手入门学习:详细介绍接口分隔原则(ISP)及其作用、代码示例 | 使用多个专门的接口,而不使用单一的总接口,降低类之间的耦合度。 |
| 【Python】新手入门学习:详细介绍组合/聚合复用原则(CARP)及其作用、代码示例 | 尽量使用合成/聚合的方式达到复用,减少继承的使用。 |
| 【Python】新手入门学习:详细介绍迪米特原则(LoD)及其作用、代码示例 | 一个对象应当对其他对象保持最少的了解,降低类之间的耦合度。 |
🌵文章目录🌵
- 🎯一、什么是依赖倒置原则(DIP)?
- 🔧二、DIP原则的作用
- 📚三、如何实现DIP原则
- 🔍四、DIP原则在实际项目中的应用
- 💡五、违反DIP原则的后果
- 🎉六、总结
🎯一、什么是依赖倒置原则(DIP)?
依赖倒置原则(Dependency Inversion Principle,简称DIP)是面向对象设计原则中的一条重要原则。其核心思想是:要依赖于抽象,不要依赖于具体实现。换言之,高层模块不应该依赖于低层模块,它们都应该依赖于抽象;抽象不应该依赖于细节,细节应该依赖于抽象。
DIP原则鼓励我们在编写代码时,将抽象层与具体实现层进行分离,通过接口或抽象类来定义抽象层,而将具体的实现细节放在具体实现层中。这样,当具体实现发生变化时,只要接口或抽象类保持不变,高层模块就不会受到影响,提高了代码的可维护性和可扩展性。
🔧二、DIP原则的作用
DIP原则在软件设计中发挥着至关重要的作用,主要体现在以下几个方面:
-
降低耦合度:通过抽象层与具体实现层的分离,降低了模块之间的耦合度。当具体实现发生变化时,只需要修改具体实现层,而不需要修改高层模块,降低了代码的维护成本。
-
提高代码的可扩展性:由于高层模块依赖于抽象层,因此可以通过添加新的具体实现来扩展系统的功能,而无需修改现有代码。
-
促进代码复用:通过定义清晰的接口或抽象类,不同的模块可以共享相同的抽象层,从而实现代码的复用。
-
增强系统的灵活性:由于高层模块不直接依赖于具体实现,因此可以轻松地替换或升级具体实现,而不会影响到整个系统的运行。
📚三、如何实现DIP原则
要实现DIP原则,我们可以采取以下几个关键步骤:
-
定义抽象层:通过接口或抽象类来定义抽象层,明确高层模块所需的功能和行为。
-
编写具体实现:实现具体的类,这些类将继承自抽象类或实现接口,提供具体的功能实现。
-
高层模块依赖于抽象:在高层模块中,只引用抽象层,而不直接引用具体实现类。这样,高层模块就可以通过抽象层来调用具体实现的功能。
下面是一个简单的Python代码示例,展示了如何实现DIP原则:
from abc import ABC, abstractmethod# 定义抽象层
class Shape(ABC):@abstractmethoddef area(self):pass# 具体实现
class Circle(Shape):def __init__(self, radius):self.radius = radiusdef area(self):return 3.14 * self.radius ** 2class Rectangle(Shape):def __init__(self, width, height):self.width = widthself.height = heightdef area(self):return self.width * self.height# 高层模块依赖于抽象
def calculate_total_area(shapes):total_area = 0for shape in shapes:total_area += shape.area()return total_area# 使用示例
shapes = [Circle(5), Rectangle(10, 20)]
total_area = calculate_total_area(shapes)
print(f"Total area: {total_area}")
在这个例子中:
-
抽象层定义:
Shape类是一个抽象基类,它定义了一个抽象的area方法。 -
具体实现:
Circle和Rectangle类是Shape的具体实现,它们分别实现了area方法,用来计算各自的面积。 -
高层模块依赖于抽象:
calculate_total_area函数是一个高层模块,它依赖于Shape抽象,而不是依赖于具体的Circle或Rectangle类。它接收一个Shape对象的列表,并调用每个对象的area方法来计算总面积。 -
细节依赖于抽象:
Circle和Rectangle类作为细节,它们实现了Shape抽象中定义的area方法。
这样的设计使得 calculate_total_area 函数可以独立于具体的形状类,只要它们实现了 Shape 抽象中定义的接口。如果将来需要添加新的形状类,只要这个新类也实现了 Shape 接口,calculate_total_area 函数就可以无缝地与之集成。因此,这个代码示例遵循了依赖倒置原则,使得系统更加灵活、可维护和可扩展。
🔍四、DIP原则在实际项目中的应用
在实际项目中,DIP原则的应用广泛而重要。它常常与工厂模式、策略模式等设计模式一起使用,以实现代码的灵活性和可扩展性。
例如,在一个复杂的业务系统中,我们可能需要处理多种不同类型的支付方式(如支付宝、微信支付、银行卡支付等)。每种支付方式都有自己的支付逻辑和接口。这时,我们可以定义一个抽象的 Payment 接口,然后为每种支付方式实现一个具体的类。业务逻辑层则依赖于 Payment 接口,而不是具体的支付实现类。这样,当需要添加新的支付方式时,我们只需要实现新的支付类,并将其注册到系统中,而无需修改现有的业务逻辑代码。
💡五、违反DIP原则的后果
如果违反了DIP原则,会导致一系列不良后果:
-
代码紧耦合:当高层模块直接依赖于具体实现时,不同模块之间会形成紧密的耦合关系。一旦具体实现发生变化,高层模块也必须进行相应的修改,这增加了代码的维护难度和成本。
-
难以扩展新功能:如果高层模块直接依赖于具体实现,那么添加新的功能或修改现有功能将变得困难。因为每次变动都需要修改高层模块的代码,这限制了系统的可扩展性。
-
测试困难:当高层模块与具体实现紧密耦合时,测试也会变得困难。因为测试人员需要模拟具体的实现细节,以验证高层模块的正确性。这增加了测试的复杂性和工作量。
-
降低了代码复用性:如果每个高层模块都直接依赖于具体的实现,那么不同的模块之间很难共享相同的抽象层。这导致了代码的重复和冗余,降低了代码的复用性。
因此,遵循DIP原则对于保持代码的健壮性、灵活性和可维护性至关重要。它有助于我们构建出易于扩展、易于测试、易于维护的软件系统。
🎉六、总结
依赖倒置原则(DIP)是面向对象设计中的重要原则之一。它强调高层模块应该依赖于抽象而不是具体实现,使得代码更加灵活、可扩展和易于维护。通过定义清晰的抽象层和具体实现层,我们可以降低模块之间的耦合度,提高代码的可复用性和可测试性。
在实际项目中,我们应该积极应用DIP原则,通过接口或抽象类来定义抽象层,将具体实现细节放在具体实现层中。这样,当具体实现发生变化时,我们只需要修改具体实现层,而无需修改高层模块,从而降低了代码的维护成本。
同时,我们也应该避免违反DIP原则,以免导致代码紧耦合、难以扩展和测试困难等问题。通过遵循DIP原则,我们可以构建出更加健壮、灵活和可维护的软件系统。
希望本文的介绍能够帮助你更好地理解依赖倒置原则,并在实际开发中灵活运用。通过不断实践和积累经验,我们可以逐渐提高软件设计的能力,创造出更加优秀和可靠的软件系统。
相关文章:
【Python】新手入门学习:详细介绍依赖倒置原则(DIP)及其作用、代码示例
【Python】新手入门学习:详细介绍依赖倒置原则(DIP)及其作用、代码示例 🌈 个人主页:高斯小哥 🔥 高质量专栏:Matplotlib之旅:零基础精通数据可视化、Python基础【高质量合集】、Py…...
嵌入式驱动学习目录索引(更新中)
前言 这是一篇索引博客,用来作为索引记录学习嵌入式Linux的过程,可以用来给自己以及需要的读者作为一个目录索引,每次更新完博客都会添加进该目录中。 嵌入式驱动学习专栏将详细记录博主学习驱动的详细过程,未来预计四个月将高强度…...
ruoyi-vue插件集成websocket
链接:插件集成 | RuoYi WebSocketServer.java:补充代码 /*** 此为广播消息* param message 消息内容*/public void sendAllMessage(String message) {LOGGER.info("【websocket.sendAllMessage】广播消息:"message);try {for(String sessionI…...
华为ce12800交换机m-lag(V-STP模式)配置举例
配置## 标题思路 采用如下的思路配置M-LAG双归接入IP网络: 1.在Switch上配置上行接口绑定在一个Eth-Trunk中。 2.分别在SwitchA和SwitchB上配置V-STP、DFS Group、peer-link和M-LAG接口。 3.分别在SwitchA和SwitchB上配置LACP M-LAG的系统优先级、系统ID。 4.分别在…...
STM32第九节(中级篇):RCC——时钟树讲解(第一节)
目录 前言 STM32第九节(中级篇):RCC——时钟树讲解 时钟树主系统时钟讲解 HSE时钟 HSI时钟 锁相环时钟 系统时钟 SW位控制 HCLK时钟 PCLKI时钟 PCLK2时钟 RTC时钟 MCO时钟输出 6.2.7时钟安全系统(CSS) 小结 前言 从…...
c/c++字符串处理标准库 string 介绍
c语言中string.h介绍 C语言的标准库中包含了一个头文件 <string.h>,该头文件提供了一系列字符串处理函数的声明和定义。以下是一些常用的函数: 字符串复制:strcpy(dest, src)。将源字符串 src 复制到目标字符串 dest,包括…...
HarmonyOS NEXT应用开发之深色模式适配
介绍 本示例介绍在开发应用以适应深色模式时,对于深色和浅色模式的适配方案,采取了多种策略如下: 固定属性适配:对于部分组件的颜色属性,如背景色或字体颜色,若保持不变,可直接设定固定色值或…...
Go微服务: 基于Go Micro框架实现微服务调用
Go Micro 1 )概述 在具体的项目开发过程中,开发者聚焦的是业务逻辑的开发和功能的实现大量的环境配置,调试搭建等基础性工作会耗费相当一部分的精力因此有必要将微服务架构中所涉及到的,相关的解决方案做集中管理和维护Go Micro …...
大模型prompt提示词如何调优?
当使用大型模型(如GPT-3.5)时,可以通过优化提示(prompt)来引导模型生成更加符合预期的内容。以下是一些调优提示词的建议: 1、清晰的问题陈述:确保你的问题或提示清晰、简明,能够准…...
【Python/crawl】如何使用Python爬虫将一系列网页上的同类图片下载到本地
【需求】 从网页https://www.zhainq.com/%e7%be%8e%e5%a5%b3%e5%86%99%e7%9c%9f%e6%9c%ba%e6%9e%84/%e6%97%a5%e6%9c%ac%e7%be%8e%e5%a5%b3%e5%86%99%e7%9c%9f/109012.html 开始,有十七页,每页都有大漂亮“小濑田麻由”的若干图片,想要将其…...
Postgresql 连接数查看,死锁问题解决
-- 查看所有连接 select * -- datname,pid,application_name,state from pg_stat_activity; -- 查询最大连接数 select max_conn-now_conn as resi_conn from (select setting::int8 as max_conn,(select count(*) from pg_stat_activity) as now_conn from pg_settings where…...
ssm蛋糕甜品商城系统(程序+文档+数据库)
** 🍅点赞收藏关注 → 私信领取本源代码、数据库🍅 本人在Java毕业设计领域有多年的经验,陆续会更新更多优质的Java实战项目,希望你能有所收获,少走一些弯路。🍅关注我不迷路🍅** 一、研究背景…...
算法空间复杂度计算
目录 空间复杂度定义 影响空间复杂度的因素 算法在运行过程中临时占用的存储空间讲解 例子 斐波那契数列递归算法的性能分析 二分法(递归实现)的性能分析 空间复杂度定义 空间复杂度(Space Complexity)是对一个算法在运行过程中临时占用存储空间大…...
C++ lambda函数个人理解
及方便自己在函数内部定义函数 int main() {int i 1;auto c [](int a, int c) {return ab;};int d a(2, i);cout<<c;return 0; }格式: auto functionname [capture](parameters) -> return_type { /* … */ }; (1)[capture] &a…...
SwiftUI的context Menu
SwiftUI的 context Menu 现在来演示一下如何使用 SwiftUI 的 Context Menu 。 代码: import SwiftUIstruct ContextMenuBootCamp: View {State var bgColor: Color .purplevar body: some View {VStack(alignment: .leading, spacing: 10.0) {Image(systemName: …...
【数据结构】树与堆 (向上/下调整算法和复杂度的分析、堆排序以及topk问题)
文章目录 1.树的概念1.1树的相关概念1.2树的表示 2.二叉树2.1概念2.2特殊二叉树2.3二叉树的存储 3.堆3.1堆的插入(向上调整)3.2堆的删除(向下调整)3.3堆的创建3.3.1使用向上调整3.3.2使用向下调整3.3.3两种建堆方式的比较 3.4堆排…...
安装CDH平台的服务器磁盘满了,磁盘清理过程记录
1.使用hdfs命令查看哪个文件占用最大 hdfs dfs -du -h /tmp 2.我的服务器上显示/tmp/hive/hive文件夹下的,一串字符串命名的文件特别大几乎把磁盘占满了 网上查到/tmp文件是临时文件,由于hiveserver2任务运行异常导致缓存未删除,正常情况下…...
《互联网的世界》第七讲-能源
本想聊聊 tcp 和 quic,但这些都属于术的范畴,变化多端,等孩子们长大了又不知变成什么样子了,趁这段时间在家,还是得讲一些相对不变的东西,或法或势。 从 安阳卖血糕的精巧篦子 想到如何做圆米粉和圆面条&a…...
前端代码整洁与规范之CSS篇
一、代码整洁 1. 命名规范 CSS 类名的命名应该简洁清晰,能够准确描述元素的作用。避免使用无意义的名称,例如“a”、“b”等,而应该使用有意义的英文单词或单词缩写。同时,也要避免使用驼峰命名法和下划线命名法混杂使用&#x…...
在【IntelliJ IDEA】中配置【Tomcat】【2023版】【中文】【图文详解】
作为一款功能强大的集成开发环境(IDE),IntelliJ IDEA为Web服务器提供了卓越的支持,从而极大地简化了程序员在Web开发过程中的工作流程。学习Java Web开发实质上就是掌握如何创造动态Web资源,这些资源在完成开发后&…...
后进先出(LIFO)详解
LIFO 是 Last In, First Out 的缩写,中文译为后进先出。这是一种数据结构的工作原则,类似于一摞盘子或一叠书本: 最后放进去的元素最先出来 -想象往筒状容器里放盘子: (1)你放进的最后一个盘子(…...
日语AI面试高效通关秘籍:专业解读与青柚面试智能助攻
在如今就业市场竞争日益激烈的背景下,越来越多的求职者将目光投向了日本及中日双语岗位。但是,一场日语面试往往让许多人感到步履维艰。你是否也曾因为面试官抛出的“刁钻问题”而心生畏惧?面对生疏的日语交流环境,即便提前恶补了…...
蓝牙 BLE 扫描面试题大全(2):进阶面试题与实战演练
前文覆盖了 BLE 扫描的基础概念与经典问题蓝牙 BLE 扫描面试题大全(1):从基础到实战的深度解析-CSDN博客,但实际面试中,企业更关注候选人对复杂场景的应对能力(如多设备并发扫描、低功耗与高发现率的平衡)和前沿技术的…...
React Native在HarmonyOS 5.0阅读类应用开发中的实践
一、技术选型背景 随着HarmonyOS 5.0对Web兼容层的增强,React Native作为跨平台框架可通过重新编译ArkTS组件实现85%以上的代码复用率。阅读类应用具有UI复杂度低、数据流清晰的特点。 二、核心实现方案 1. 环境配置 (1)使用React Native…...
Nuxt.js 中的路由配置详解
Nuxt.js 通过其内置的路由系统简化了应用的路由配置,使得开发者可以轻松地管理页面导航和 URL 结构。路由配置主要涉及页面组件的组织、动态路由的设置以及路由元信息的配置。 自动路由生成 Nuxt.js 会根据 pages 目录下的文件结构自动生成路由配置。每个文件都会对…...
第一篇:Agent2Agent (A2A) 协议——协作式人工智能的黎明
AI 领域的快速发展正在催生一个新时代,智能代理(agents)不再是孤立的个体,而是能够像一个数字团队一样协作。然而,当前 AI 生态系统的碎片化阻碍了这一愿景的实现,导致了“AI 巴别塔问题”——不同代理之间…...
DBAPI如何优雅的获取单条数据
API如何优雅的获取单条数据 案例一 对于查询类API,查询的是单条数据,比如根据主键ID查询用户信息,sql如下: select id, name, age from user where id #{id}API默认返回的数据格式是多条的,如下: {&qu…...
【论文阅读28】-CNN-BiLSTM-Attention-(2024)
本文把滑坡位移序列拆开、筛优质因子,再用 CNN-BiLSTM-Attention 来动态预测每个子序列,最后重构出总位移,预测效果超越传统模型。 文章目录 1 引言2 方法2.1 位移时间序列加性模型2.2 变分模态分解 (VMD) 具体步骤2.3.1 样本熵(S…...
【数据分析】R版IntelliGenes用于生物标志物发现的可解释机器学习
禁止商业或二改转载,仅供自学使用,侵权必究,如需截取部分内容请后台联系作者! 文章目录 介绍流程步骤1. 输入数据2. 特征选择3. 模型训练4. I-Genes 评分计算5. 输出结果 IntelliGenesR 安装包1. 特征选择2. 模型训练和评估3. I-Genes 评分计…...
Reasoning over Uncertain Text by Generative Large Language Models
https://ojs.aaai.org/index.php/AAAI/article/view/34674/36829https://ojs.aaai.org/index.php/AAAI/article/view/34674/36829 1. 概述 文本中的不确定性在许多语境中传达,从日常对话到特定领域的文档(例如医学文档)(Heritage 2013;Landmark、Gulbrandsen 和 Svenevei…...
