大话设计模式-依赖倒转原则

news/2024/9/23 17:23:45/

依赖倒转原则

在大话设计模式这本书中,作者通过电话修电脑这个例子引入了面向对象设计的基本原则之一:依赖倒转原则

概念

依赖倒转原则是面向对象设计的基本原则之一,它用于减少类之间的耦合,提高系统的灵活性和可维护性。在书中,依赖倒转原则的原话解释是抽象不应该依赖细节,细节应该依赖于抽象。它的主要内容可以分为两个部分:

  1. 高层模块不应该依赖低层模块。两者都应该依赖其抽象。
  2. 抽象不应该依赖于细节。细节应该依赖于抽象。
    简单点说,就是我们在编程的时候要依赖于抽象(接口或抽象类),不要依赖于具体的类(对象)。

这一原则与我们前文中所讲解的里氏代换原则相辅相成,当我们满足里氏代换原则时,子类能够完全替换父类。而满足依赖倒转原则时,细节依赖于抽象(抽象类一般是父类)。因此当同时遵循这两个原则的时候,我们的代码就能够通过子类灵活的进行扩展。

例子

光讲上面的概念可能难以理解其中的含义,这里举一个具体的例子。

  • 假设我们正在开发一个新闻应用,这个应用有一个NewsService类,它负责从不同的新闻源获取新闻。一开始,我们只从网络获取新闻,所以我们可能会有以下的设计:
class NewsService {NetworkNewsFetcher fetcher;NewsService() {this.fetcher = new NetworkNewsFetcher();}List<News> getNews() {return fetcher.fetch();}
}class NetworkNewsFetcher {List<News> fetch() {// fetch news from the network}
}

在这个设计中,NewsService直接依赖于NetworkNewsFetcher,这意味着如果我们想从其他来源(如本地文件)获取新闻,我们就需要修改NewsService的代码。
现在,让我们按照依赖倒转原则来重新设计这个系统:

interface NewsFetcher {List<News> fetch();
}class NewsService {NewsFetcher fetcher;NewsService(NewsFetcher fetcher) {this.fetcher = fetcher;}List<News> getNews() {return fetcher.fetch();}
}class NetworkNewsFetcher implements NewsFetcher {@Overridepublic List<News> fetch() {// fetch news from the network}
}class LocalNewsFetcher implements NewsFetcher {@Overridepublic List<News> fetch() {// fetch news from a local file}
}
  • 在这个新的设计中,NewsService依赖于NewsFetcher接口,而不是具体的NetworkNewsFetcher类。这样,我们就可以在不修改NewsService的代码的情况下,通过添加新的NewsFetcher实现(如LocalNewsFetcher)来扩展系统的功能。这个例子是通过代码来解释依赖倒转原则,能够帮助有一定编程基础的同学能够更加清晰地体会到依赖倒转原则的优点。

优缺点

在大话设计模式中,有这样一句话:可以把PC电脑理解成是大的软件系统,任何部件如CPU、内存、硬盘、显卡等都可以理解为程序中封装的类或程序集,由于PC易插拔的方式,那么不管哪一个出问题,都可以在不影响别的部件的前提下进行修改或替换。

在这句话中 PC电脑依赖于CPU、内存、硬盘、显卡等硬件设备,而这些硬件损坏的时候,我们能够很方便的对其进行更换,这就依赖于PC易插拔的方式。我们可以思考一个这样的问题,PC易插拔的方式是怎么做到的?其实就是因为它遵循了依赖倒置原则,严格意义上来讲,PC电脑所依赖的并不是具体的某一块CPU、内存、硬盘、显卡等硬件设备,而是依赖能够插进主板插槽的硬件设备。而这个主板预留出的插槽其实就是我们前面所谓的“抽象”(接口或抽象类),只有主板依赖于插槽,这些硬件的生产厂商,再根据插槽的形状去做对应的硬件,才能够实现PC易插拔。否则, 如果PC依赖的具体的某一块CPU、内存、硬盘、显卡等硬件设备,那如果硬件设备坏了,就得整个PC都换掉,因为PC只“用得惯”这一块硬件。

因此,总结一下依赖倒转原则的优点:它使我们的代码更加灵活,更容易扩展和维护。
在这里插入图片描述


http://www.ppmy.cn/news/1426080.html

相关文章

EC20-4G模块使用AT指令分析设备网络异常原因

1&#xff1a;现象描述 实际使用4G模块的过程中&#xff0c;我们可能遇到模块无法联网的现象&#xff0c;不清楚具体的流程&#xff0c;没办法分析不同的场景问题&#xff1b;下面的流程帮助大家分析4G模块的联网流程&#xff0c;如果网络不通的话&#xff0c;应该怎么一步步排…

Spring Boot | Spring Boot 应用的 “打包” 和 “部署”

目录: Spring Boot 应用的 “打包” 和 “部署” :一、Jar包方式打包部署 ( SpringBoot默认以 "Jar包" 形式进行 “打包部署” ) :1.1 "Jar包" 方式 “打包” :① 添加Maven “打包插件”② 使用IDEA开发工具进行 "打包" 1.2 "Jar包" …

【LeetCode热题100】【动态规划】单词拆分

题目链接&#xff1a;139. 单词拆分 - 力扣&#xff08;LeetCode&#xff09; 看能不能用字符串列表里面的字符串组成这个字符串&#xff0c;可以反复使用 即完全背包问题&#xff0c;同之前的完全平方数、零钱兑换&#xff0c;相当于给定几个数&#xff0c;可以反复用&#…

【JavaSE】JDK17的一些特性

前言 从springboot3.0开始&#xff0c;已经不⽀持JDK8了 选⽤Java17&#xff0c;概括起来主要有下⾯⼏个主要原因 JDK17是LTS(⻓期⽀持版)&#xff0c;可以免费商⽤到2029年。⽽且将前⾯⼏个过渡版&#xff08;JDK9-JDK16&#xff09; 去其糟粕&#xff0c;取其精华的版本JDK17…

阿里云服务器2024年十大优惠活动,手动整理建议收藏!

2024年阿里云服务器优惠活动大全包括&#xff1a;云服务器新人特惠、云小站、阿里云免费中心、学生主机优惠、云服务器精选特惠、阿里云领券中心等&#xff0c;活动上阿里云服务器ECS经济型e实例2核2G、3M固定带宽99元一年、ECS u1云服务器2核4G5M带宽199元一年&#xff0c;轻量…

探索ERC20代币:构建您的第一个去中心化应用

下面文章中会涉及到该资源中的代码&#xff0c;如果想要完整版代码可以私信我获取&#x1f339; 文章目录 概要整体架构流程技术名词解释ERC20智能合约web3.js 技术细节ERC20合约部署创建前端界面前端与智能合约互连运行DAPP 小结 概要 在加密货币世界中&#xff0c;ERC20代币…

Pytest精通指南(14)Parametrize之indirect(间接参数)

文章目录 官方概念概念分析官方示例示例分析验证indirect为True但不指定fixture验证indirect为True但不存在fixture 官方概念 概念分析 在pytest的pytest.mark.parametrize装饰器中&#xff0c;indirect参数用于指示是否应该从fixtures中解析参数值&#xff0c;而不是直接使用提…

Qt gsl库配置踩坑记录

想求解非线性方程组&#xff0c;之前使用拟牛顿法写过相关的matlab代码&#xff0c;这次想移植到C代码&#xff0c;网上说gsl库挺好用的&#xff0c;于是我也想试一下。相关参考&#xff1a; 【C】GSL(GNU Scientific Library) 的安装及在 Visual Studio 2017 中的使用 QT5使用…