软件设计原则之依赖倒置原则

news2024/9/20 18:52:01

依赖倒置原则(Dependency Inversion Principle, DIP)是软件设计中一个非常重要的原则,它属于面向对象设计的SOLID原则之一。这个原则的核心在于通过抽象来降低模块间的耦合度,使得系统更加灵活和可维护。

目录

  • 依赖倒置原则的基本思想
  • 依赖倒置原则的好处
  • 依赖倒置原则的实现方式
  • 打个比方
  • 不遵守的例子
  • Demo

依赖倒置原则的基本思想

高层模块不应该依赖低层模块,两者都应该依赖其抽象:这意味着在软件系统中,设计上层模块(如业务逻辑层)时,不应该直接依赖于下层模块(如数据访问层)的具体实现。相反,它们都应该依赖于一个共同的抽象层(如接口或抽象类)。
抽象不应该依赖细节,细节应该依赖抽象: 抽象层(接口或抽象类)定义了高层模块和低层模块之间的通信规范,而不涉及具体的实现细节。具体的实现(细节)则依赖于这些抽象规范来实现具体的功能。

依赖倒置原则的好处

降低耦合度: 通过依赖抽象层而不是具体实现,可以减少模块间的直接依赖,从而降低系统的耦合度。
提高灵活性: 由于高层模块不直接依赖于低层模块的具体实现,因此可以更容易地更换或升级低层模块,而无需修改高层模块。
增强可测试性: 依赖倒置原则使得单元测试更加容易,因为可以通过模拟(mocking)或存根(stubbing)抽象层来测试高层模块。

依赖倒置原则的实现方式

接口和抽象类的使用: 定义清晰的接口或抽象类,并在高层模块和低层模块之间使用这些接口或抽象类进行通信。
依赖注入: 通过依赖注入(Dependency Injection, DI)技术,将低层模块的实现注入到高层模块中,而不是在高层模块中直接创建低层模块的对象。依赖注入不同于依赖倒置,因为依赖性倒置是为软件模块定义一个抽象策略,而依赖注入是依赖倒置的一种实现方式。

打个比方

在这里插入图片描述
正如你可以看到在上面的图像,我们有一个端口为每个外部设备,我可以关联一个外部设备,并做我们的工作。
但问题是,我不能将我的键盘连接到打印机端口,反之亦然。其他设备也会出现同样的问题。这就像一个紧密耦合,我不能在给定的接口上更改我的外部设备。
如果我有一个 USB 接口,那么我可以很容易地连接任何设备到我的机器和执行我的任务。
在这里插入图片描述

不遵守的例子

在这里插入图片描述

Public Class Customer  
{  
    CustomerRepository CustomerRepository;  
    Public Customer  
    {  
        CustomerRepository = new CustomerRpository();  
    }  

    Public bool Save()  
    {  
        CustomerRepository.Save();  
    }  
}  

Public class CustomerRepository  
{  
    Public bool Save(dattype data)  
    {  
              //Sql Connection object and Save data in Sql server   
    }  
}

上面的代码是紧密耦合的,因为当前存储库处理 SQL 服务器。因此,如果需求是使用 Oracle 服务器,则需要对 Customer 类进行修改。
因此,为了避免这种情况,应该让客户类依赖于抽象。下面是一个类图,其中客户依赖于抽象并支持 SQL 和 Oracle 服务器。
在这里插入图片描述
对于软件中设计的模块也是一样的,因为依赖性反转是关于提供一组抽象策略,其细节依赖于这组抽象策略,以及在软件系统中提供灵活性的策略。
应用程序模块高度耦合,这将导致模块的可测试性变得困难,模块的并行开发变得困难,在模块中进行修改以及所依赖的模块中进行更改时,需要进行许多更改。

Demo

假设我们有一个日志系统,其中有一个ILogger接口和一个Application类。Application类需要记录日志,但它不直接依赖于具体的日志实现类(比如ConsoleLogger或FileLogger),而是依赖于ILogger接口。

public interface ILogger  
{  
    void Log(string message);  
}

然后,我们创建两个具体的日志实现类:ConsoleLogger和FileLogger,它们都实现了ILogger接口:

public class ConsoleLogger : ILogger  
{  
    public void Log(string message)  
    {  
        Console.WriteLine(message);  
    }  
}  
  
public class FileLogger : ILogger  
{  
    private string filePath;  
  
    public FileLogger(string filePath)  
    {  
        this.filePath = filePath;  
    }  
  
    public void Log(string message)  
    {  
        // 这里只是模拟写入文件,实际项目中应该使用文件IO操作  
        Console.WriteLine($"Logging to file: {filePath} - Message: {message}");  
    }  
}

接下来,我们定义Application类,它依赖于ILogger接口而不是具体的日志实现类。我们将通过构造函数注入的方式将日志记录器的依赖注入到Application类中:

public class Application  
{  
    private ILogger logger;  
  
    // 构造函数注入ILogger依赖  
    public Application(ILogger logger)  
    {  
        this.logger = logger;  
    }  
  
    public void Run()  
    {  
        // 执行一些业务逻辑  
        string message = "This is an important message.";  
        logger.Log(message);  
  
        // 更多的业务逻辑...  
    }  
}

最后,在客户端代码中,我们根据需要创建具体的日志实现类,并将其注入到Application类的实例中:

class Program  
{  
    static void Main(string[] args)  
    {  
        // 使用ConsoleLogger  
        ILogger consoleLogger = new ConsoleLogger();  
        Application appWithConsoleLogger = new Application(consoleLogger);  
        appWithConsoleLogger.Run();  
  
        // 或者使用FileLogger  
        ILogger fileLogger = new FileLogger("app.log");  
        Application appWithFileLogger = new Application(fileLogger);  
        appWithFileLogger.Run();  
    }  
}

在这个例子中,Application类是高层模块,它依赖于ILogger接口这一抽象层,而不是依赖于具体的日志实现类(ConsoleLogger或FileLogger)。这种设计方式使得我们可以轻松地更换日志实现,而无需修改Application类的代码,从而实现了依赖倒置。同时,这种设计也提高了系统的可测试性,因为我们可以通过模拟(mocking)ILogger接口来测试Application类的行为。

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

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

相关文章

对标GPT-4o,科大讯飞正以大模型重塑语音产业

每个科技时代,都有每个时代的“入口”和“推手”。 在PC时代,浏览器和搜索引擎是主要入口,用户通过键盘和鼠标进行交互。移动互联时代,APP和应用商店成为典型入口,用户用手指和触摸屏进入互联网世界。而在眼下的AI时代…

8月27c++

提示并输入一个字符串&#xff0c;统计字符串中字母、数字、空格和其他字符的个数 代码 #include <iostream> #include <cstring> using namespace std;int main() {string str;cout<<"输入一个字符串";getline(cin,str);//输入字符串int lenstr…

【vulhub】Weblogic WLS Core Components 反序列化命令执行漏洞(CVE-2018-2628)

简单来说就是先用序列化工具ysoserial启动一个JRMP服务&#xff0c;加载先相关漏洞利用链&#xff0c;加载你要执行的恶意代码。 并将上述结果通过序列化工具ysoserial将我们的恶意代码进行一个序列化操作。 第二步就是将我们的exp去加载ysoserial序列化后的数据&#xff0c;后…

vue侧边栏

在Vue中创建一个侧边栏&#xff08;Sidebar&#xff09;是一个常见的需求&#xff0c;特别是在构建管理界面或需要导航菜单的应用时。侧边栏通常用于展示应用的导航链接或菜单项&#xff0c;用户可以通过点击这些链接来访问应用的不同部分。 <template><el-tree :data…

openEuler安装Docker和踩坑分析

我是用的openEuler版本&#xff1a;20.03 LTS SP1&#xff0c;安装在虚拟机中&#xff0c;使用ssh连接 在openEuler上安装Docker还是让我踩了不少坑&#xff0c;先看看这些坑是如何产生的 虽然官方没有说openEuler是基于什么开源系统开发的&#xff0c;但大致内容和CentOS相似…

【教学新纪元】大学电工电子课堂大变身!SmartEDA电路仿真软件助你高效授课✨

在快速迭代的科技时代&#xff0c;教学方式的创新成为了提升教学质量的关键。作为一位深耕电工电子原理教学领域的老师&#xff0c;你是否曾梦想过将复杂的电路理论以直观、互动的方式展现给学生&#xff1f;今天&#xff0c;就让我们一起探索如何利用SmartEDA电路仿真软件&…

鸿蒙开发5.0【基于lycium的开源c库编译与集成】

场景描述 对于c库编译问题&#xff0c;应用经常会遇到如下业务诉求 场景一&#xff1a;基于HarmonyOS编译开源C库 场景二&#xff1a;开源c库编译完成后的集成 方案描述 场景一&#xff1a; 需要使用开源c库 lycium的使用说明&#xff1a;lycium的特点就是自动化编译&am…

8.28-回顾+容器与主机之间的通信+跨主机容器之间的通信

一、回顾 1.启动docker systemctl start docker 2.拉取registry docker pull registry 3.启动镜像&#xff0c;同时挂载目录&#xff08;保存镜像&#xff09;端口映射5000 docker run -d -v /regist/:/var/lib/registry/ -p5000:5000 registry:latest 4.修改/etc/docker/d…

培训第三十八天(上传镜像,私有仓库下载镜像,跨主机容器间的通信,harbor软件包下载)

1、harbor软件包下载 https://github.com/search?qharbor&typerepositories 2、出现拒绝连接错误&#xff0c;可能是由于容器没开 # 问题解决&#xff1a;[rootdocker ~]# curl localhost:5000/v2/_catalogcurl: (7) Failed connect to localhost:5000; 拒绝连接[rootdoc…

pdf怎么转换成excel?掌握好这9个pdf转换方法就够了(全)

pdf怎么转换成excel&#xff1f;日常的办公生活中&#xff0c;我们经常需要接触很多文档格式&#xff0c;而pdf格式文件因为其稳定性和安全性受到很多办公人士的喜爱。但PDF文件不能直接编辑&#xff0c;很多小伙伴们就会出现关于pdf格式转换的难题&#xff0c;比如说想把一份带…

MATLAB虫害检测预警系统

一、课题介绍 本课题是基于MATLAB颜色的植物虫害检测识别&#xff0c;可以辨析植物叶子属于是轻度虫害&#xff0c;中度虫害&#xff0c;严重虫害&#xff0c;正常等四个级别。算法流程&#xff1a;每种等级叶子分别放在同一个文件夹&#xff0c;训练得到每个文件夹每个叶…

SSL安全认证网关:保障网络安全的强大护盾

随着信息技术的飞速发展&#xff0c;我们的生活和工作越来越依赖于网络&#xff0c;但与此同时&#xff0c;网络安全威胁也日益严峻。为了保护我们的信息安全&#xff0c;各种安全技术和产品应运而生&#xff0c;其中SSL安全认证网关就是一种非常重要的安全防护工具。 今天&…

3款伪原创工具,为你轻松一键生成原创文案

在当今信息爆炸的时代&#xff0c;原创内容的重要性愈发凸显。然而&#xff0c;对于许多创作者来说&#xff0c;创作原创文案却是一项费时费力的挑战。幸运的是&#xff0c;随着科技的进步&#xff0c;现在有三款伪原创工具能够帮助你轻松一键生成原创文案&#xff0c;为你节省…

DNS服务器的配置(服务名named,端口53)

目录 前言 配置文件 DNS服务器的配置 主配置文件 扩展配置文件 区域配置文件 重启服务 配置防火墙 配置客户端dns 前言 DNS服务器的主要作用是将人类可读的域名转换为机器可读的IP地址&#xff0c;从而方便用户访问互联网资源。 在互联网中&#xff0c;设备需要通过I…

基于资源管控+TiCDC实现多业务融合容灾测试

作者&#xff1a; 数据源的TiDB学习之路 原文来源&#xff1a; https://tidb.net/blog/959b8d07 背景 金融机构越来越多的选择将多套业务系统融合到一套分布式数据库集群来支撑&#xff0c;一方面可以节约硬件成本&#xff0c;另一方面可以简化运维复杂性。多租户能力及资源…

【电控笔记z26】串级PID单环位置PID

1P-PI 传函(梅森法) : 2PI-P 3PID 三者等效

HyperMesh概述与有限元分析简介

1.1 HyperMesh 概述 本节将介绍有限单元法基本原理&#xff0c;HyperMesh 软件基本功能及界面介绍&#xff0c;获取在线帮助等内容。 1.1.1 有限元分析方法简介 有限单元法&#xff08;FEM&#xff09;是一种可以精确预测复杂结构在外界载荷作用下响应的方法&#xff0c;该数…

问界都回暖了,是谁还在持续掉队?

文/王俣祺 导语&#xff1a;在8月份的最后一个完整周&#xff0c;国内汽车市场的销量表现全面提升&#xff0c;乘用车市场销量达到了46.6万辆车&#xff0c;环比增长13.1%。其中&#xff0c;新能源汽车销量达到24.2万辆&#xff0c;环比增长11.6%&#xff0c;市场渗透率达到了…

《探索现代JavaScript中的异步编程》

探索现代JavaScript中的异步编程 随着Web应用变得越来越复杂&#xff0c;前端开发中对异步处理的需求也日益增加。JavaScript 作为 Web 开发中最主要的语言之一&#xff0c;提供了多种异步编程的方法来帮助开发者编写高效、可维护的应用程序。本文将介绍几种现代 JavaScript 中…

P5928 [国家集训队] 文学 题解

Description 给定 n n n 个半平面 a i x b i y ≤ c i a_i xb_i y\le c_i ai​xbi​y≤ci​ 和 p p p 个关键点 ( x i , y i ) (x_i,y_i) (xi​,yi​)&#xff0c;第 i i i 个半平面有价格 w i w_i wi​&#xff0c;你需要选择一些半平面覆盖所有的关键点&#xff0c;同…