设计模式:依赖倒转原则 - 依赖抽象,解耦具体实现

一、为什么用依赖倒转原则?

在软件开发中,类与类之间的依赖关系是架构设计中的关键。如果依赖过于紧密,系统的扩展性和维护性将受到限制。为了应对这一挑战,依赖倒转原则(Dependency Inversion Principle,DIP)应运而生。它旨在通过解耦高层模块与低层模块,从而提升系统的灵活性和可维护性。

依赖倒转原则的核心思想:

• 高层模块不应依赖低层模块,二者都应依赖于抽象。

• 抽象不应依赖细节,细节应依赖抽象。

通过遵循这一原则,能够降低模块之间的耦合度。当需求变更时,影响的范围更小,修改更加安全。

二、依赖倒转原则实例

示例代码如下:通过对比来看,依赖倒转原则的优势。

场景:一个用户管理系统,系统需要进行邮件通知。

1.反例:不遵守依赖倒转原则

低层模块 EmailService

// 低层模块:邮件发送功能
public class EmailService {public void sendEmail(String message) {System.out.println("Sending email: " + message);}
}

高层模块 UserManager

// 高层模块:用户管理
public class UserManager {private EmailService emailService;public UserManager() {// 高层模块直接依赖低层模块this.emailService = new EmailService(); }public void registerUser(String username) {System.out.println("Registering user: " + username);emailService.sendEmail("Welcome to the system, " + username);}
}

缺点:

UserManager类直接依赖了EmailService类,EmailService的变化会影响到UserManager,UserManager的功能无法灵活变化。

2.正例:遵守依赖倒转原则

第1步:抽象接口 NotificationService

// 抽象接口:通知服务
public interface NotificationService {void sendNotification(String message);
}

第2步:低层模块 EmailService

// 低层模块:邮件发送功能(具体实现)
public class EmailService implements NotificationService {@Overridepublic void sendNotification(String message) {System.out.println("Sending email: " + message);}
}

第3步:高层模块 UserManager

// 高层模块:用户管理
public class UserManager {private NotificationService notificationService;// 构造函数,接收 NotificationService 类型的抽象public UserManager(NotificationService notificationService) {this.notificationService = notificationService;}// 用户注册功能public void registerUser(String username) {System.out.println("Registering user: " + username);notificationService.sendNotification("Welcome to the system, " + username);}
}

第4步:测试类 Main

// 测试用的 Main 方法
public class Main {public static void main(String[] args) {// 创建具体的底层模块:邮件发送服务NotificationService emailService = new EmailService();// 创建高层模块:用户管理,并传入具体的底层模块UserManager userManager = new UserManager(emailService);// 调用用户注册功能userManager.registerUser("JohnDoe");}
}

代码说明:

• NotificationService 接口:发送通知的抽象方法,任何具体的通知服务都实现此接口。

• EmailService:底层模块,是通知服务接口的具体类。它用于处理邮件发送逻辑。

• UserManager:高层模块,它通过构造函数接受NotificationService接口的依赖,从而不直接依赖于EmailService类。

• Main 类:程序入口,创建了邮件服务类,并将其注入到用户管理类中,最后调用registerUser方法进行注册并发送邮件通知。

运行结果:

Registering user: JohnDoe
Sending email: Welcome to the system, JohnDoe

优化后的优势:

• 依赖于抽象,而非具体实现:

UserManager类依赖于NotificationService接口,而不是具体的EmailService类。

这样,如果更改通知方式(如短信通知),只需实现一个新的NotificationService接口实现类(如SmsService),并将其注入到UserManager中,而无需修改UserManager类代码。

• 提升灵活性与扩展性:

通过依赖注入,UserManager 类与 Email-Service 解耦,使得系统更易于测试和扩展,满足不同需求的变化。

三、依赖倒转原则的价值

• 降低耦合度:

高层模块不再依赖低层模块的实现细节,而是依赖于抽象,从而大大降低了模块间的耦合。

• 提高灵活性与扩展性:

系统能够轻松适应需求变化和技术变更。如新增功能或模块时(如替换邮件通知为推送通知),只需产出新的实现类, 而无需修改高层模块的代码。

• 简化测试:

通过抽象化,解耦了模块间的依赖,单元测试更容易。此外,可通过模拟(mock)替代依赖的实现,进行独立测试。

四、适用场景

以下场景都能充分发挥依赖倒转原则的价值:

• 需求变化频繁的系统

• 功能模块扩展

• 高可测试性的系统

五、总结

依赖倒转原则强调高层模块与低层模块都依赖于抽象接口,从而减少耦合、提高系统的灵活性和可扩展性。

在实际开发中,遵循这一原则能够设计出更具适应性的系统,使得需求变化和功能扩展变得更加简单和安全。

同时,它还能提升系统的可维护性和可测试性,是构建高质量软件系统的关键,尤其适用于复杂或需求变动频繁的项目。

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

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

相关文章

vue+d3js+fastapi实现天气柱状图折线图饼图

说明: vued3jsfastapi实现天气柱状图折线图饼图 效果图: step0:postman 1. 生成天气数据(POST请求):URL: http://localhost:8000/generate-data/?year2024&month3&seed42 方法: POST Headers:Content-Type:…

UE5,LogPackageName黄字警报处理方法

比如这个场景,淘宝搜索,ue5 T台,转为ue5.2后,选择物体,使劲冒错。 LogPackageName: Warning: DoesPackageExist called on PackageName that will always return false. Reason: 输入“”为空。 2. 风险很大的删除法&…

量子代理签名:量子时代的数字授权革命

1. 量子代理签名的定义与核心原理 量子代理签名(Quantum Proxy Signature, QPS)是经典代理签名在量子信息领域的延伸,允许原始签名者(Original Signer)授权给代理签名者(Proxy Signer)代为签署文…

【ESP32-C6】Base on esptool commands to enable Flash Encryption and Secure Boot

Please refer to Security Guides Security Overview Flash Encryption Secure Boot v2 Security Features Enablement Workflows Vulnerabilities You can base on “esp-idf/examples/security/flash_encryption” example for testing. Partition Table setting&#…

Kotlin 学习-方法和参数类型

/*** kotlin 的方法有三种* */fun main() {/*** 方法一* 1.普通类的成员方法申明与调用* (1)需要先构建出实例对象,才能访问成员方法* (2)实例对象的构建只需要在类名后面加上()* */Person().test()/*** 方法二&#x…

头歌 | WPS文档基本操作

若为出现预期结果可私信我答疑 2025年4月9日 第1关:新建WPS文档和保存文档 在本地创建一个1.sh,内容写入echo 我的第一个WPS文档.docx创建成功点击工具栏 点击上传文件把刚刚创建的1.sh上传 点击图形化 点击workspace>userfiles, 复制上传的文件1.sh返回上一级…

使用docker 安装向量数据库Milvus

Miluvs 官网 www.milvus.io/ https://milvus.io/docs/zh/install_standalone-docker-compose-gpu.md 一、基本概念 向量数据库:Milvus是一款云原生向量数据库,它支持多种类型的向量,如浮点向量、二进制向量等,并且可以处理大规模…

ps 人像学习

视频: 一ps快捷键 1.1 创建图层 ctrlj 1.2 放大缩小图片的大小 按住alt 滚轮 1.3 移动图片 空格 左键 1.4 撤回 ctrlz 二 精修的第一步是去除斑点,瑕疵, 2.1 污点修复画笔工具 新建一个图层,点击污点修复工具进行修复…

数据结构第五版【李春葆】

​ 数据结构教程上机实验指导第5版(李春葆主编).pdf 数据结构教程(第5版)(李春葆).pdf 数据结构教程(第五版)课后习题参考答案(李春葆).pdf 数据结构教…

(二十三)安卓开发中数据存储之Room详解

在安卓开发中,Room 是一个强大的本地数据库解决方案,它是 Android Jetpack 的一部分,基于 SQLite 构建,提供了更高层次的抽象。Room 简化了数据库操作,减少了样板代码,同时支持与 LiveData 和 ViewModel 的…

[C++面试] 初始化相关面试点深究

一、入门 1、C中基础类型的初始化方式有哪些?请举例说明 ​默认初始化​ 对于全局变量和静态变量,基础类型(如int、float、double等)会被初始化为 0;而对于局部变量,其值是未定义的,包含随机…

网络安全之-信息收集

域名收集 域名注册信息 站长之家 https://whois.chinaz.com/ whois 查询的相关网站有:中国万网域名WHOIS信息查询地址: https://whois.aliyun.com/西部数码域名WHOIS信息查询地址: https://whois.west.cn/新网域名WHOIS信息查询地址: http://whois.xinnet.com/domain/whois/in…

Linux网络http与https

应用层协议HTTP 提示 因为现在大多数都是https,所以就用https来介绍http,https比http多了一个加密功能,不影响介绍http。 什么是http 虽然我们说, 应用层协议是我们程序猿自己定的. 但实际上, 已经有大佬们定义了一些现成的, 又非常好用的…

讲解贪心算法

贪心算法是一种常用的算法思想,其在解决问题时每一步都做出在当前状态下看起来最优的选择,从而希望最终能够获得全局最优解。C作为一种流行的编程语言,可以很好地应用于贪心算法的实现。下面我们来讲一篇关于C贪心算法的文章。 目录 贪心算法…

vue3中watch的使用示例

使用情况说明: 1、父组件中有个表格,点击表格行的修改基础信息,弹出修改对话框; 2、修改内容点击确认,发送请求,后端更新数据;不修改内容不发送请求; 3、可以连续修改&#xff1b…

Spring MVC 请求类型注解详解

Spring MVC 请求类型注解详解 1. 核心注解分类 Spring MVC 中的请求处理注解分为以下几类: 类别注解示例作用范围方法级注解RequestMapping, GetMapping 等方法级别参数级注解RequestParam, RequestBody方法参数模型/会话注解ModelAttribute, SessionAttributes方…

C#: DxF文件中Spline解析

以下是使用C#解析DXF文件中Spline(样条曲线)的完整代码示例,使用流行的netDxf库来处理DXF文件: 1. 安装netDxf库 首先通过NuGet安装netDxf库: Install-Package netDxf 2. 完整Spline解析代码 using System; using System.Collections.Ge…

【软考系统架构设计师】系统架构设计知识点

1、 从需求分析到软件设计之间的过渡过程称为软件架构。 软件架构为软件系统提供了一个结构、行为和属性的高级抽象,由构件的描述、构件的相互作用(连接件)、指导构件集成的模式以及这些模式的约束组成。 软件架构不仅指定了系统的组织结构和…

二.springBoot项目集成ElasticSearch及使用

二.springBoot项目集成ElasticSearch及使用 1.依赖引入2.ElasticSearch常见用法 1.依赖引入 <!--elasticsearch搜索引擎--> <!--高版本7.0后TransportClient已被淘汰&#xff0c;用rest-high-level-client代替--> <dependency><groupId>org.elasticse…

微服务多模块构建feign项目过程与一些报错(2025详细版)

目录 1.eureka-server的注意事项 2.eureka-feign的注意事项 3.多模块构建feign项目过程 3.1创建父项目 3.2创建子项目eureka-server 3.3创建子项目eureka-provider 3.4创建子项目eureka-feign 3.5运行 给个点赞谢谢 1.eureka-server的注意事项 eureka-server的yml文件…