【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资源,这些资源在完成开发后&…...
Oracle查询表空间大小
1 查询数据库中所有的表空间以及表空间所占空间的大小 SELECTtablespace_name,sum( bytes ) / 1024 / 1024 FROMdba_data_files GROUP BYtablespace_name; 2 Oracle查询表空间大小及每个表所占空间的大小 SELECTtablespace_name,file_id,file_name,round( bytes / ( 1024 …...
前端倒计时误差!
提示:记录工作中遇到的需求及解决办法 文章目录 前言一、误差从何而来?二、五大解决方案1. 动态校准法(基础版)2. Web Worker 计时3. 服务器时间同步4. Performance API 高精度计时5. 页面可见性API优化三、生产环境最佳实践四、终极解决方案架构前言 前几天听说公司某个项…...

376. Wiggle Subsequence
376. Wiggle Subsequence 代码 class Solution { public:int wiggleMaxLength(vector<int>& nums) {int n nums.size();int res 1;int prediff 0;int curdiff 0;for(int i 0;i < n-1;i){curdiff nums[i1] - nums[i];if( (prediff > 0 && curdif…...

剑指offer20_链表中环的入口节点
链表中环的入口节点 给定一个链表,若其中包含环,则输出环的入口节点。 若其中不包含环,则输出null。 数据范围 节点 val 值取值范围 [ 1 , 1000 ] [1,1000] [1,1000]。 节点 val 值各不相同。 链表长度 [ 0 , 500 ] [0,500] [0,500]。 …...

(二)原型模式
原型的功能是将一个已经存在的对象作为源目标,其余对象都是通过这个源目标创建。发挥复制的作用就是原型模式的核心思想。 一、源型模式的定义 原型模式是指第二次创建对象可以通过复制已经存在的原型对象来实现,忽略对象创建过程中的其它细节。 📌 核心特点: 避免重复初…...

前端开发面试题总结-JavaScript篇(一)
文章目录 JavaScript高频问答一、作用域与闭包1.什么是闭包(Closure)?闭包有什么应用场景和潜在问题?2.解释 JavaScript 的作用域链(Scope Chain) 二、原型与继承3.原型链是什么?如何实现继承&a…...
什么?连接服务器也能可视化显示界面?:基于X11 Forwarding + CentOS + MobaXterm实战指南
文章目录 什么是X11?环境准备实战步骤1️⃣ 服务器端配置(CentOS)2️⃣ 客户端配置(MobaXterm)3️⃣ 验证X11 Forwarding4️⃣ 运行自定义GUI程序(Python示例)5️⃣ 成功效果 1. 启用 TCP/IP 协议 打开 “SQL Server 配置管理器”。导航到:SQL Server 网络配…...