Java依赖倒转原则_动力节点Java学院整理

发布时间 - 2026-01-11 02:43:39    点击率:

定义:高层模块不应该依赖低层模块,二者都应该依赖其抽象;抽象不应该依赖细节;细节应该依赖抽象。

问题由来:类A直接依赖类B,假如要将类A改为依赖类C,则必须通过修改类A的代码来达成。这种场景下,类A一般是高层模块,负责复杂的业务逻辑;类B和类C是低层模块,负责基本的原子操作;假如修改类A,会给程序带来不必要的风险。

解决方案:将类A修改为依赖接口I,类B和类C各自实现接口I,类A通过接口I间接与类B或者类C发生联系,则会大大降低修改类A的几率。

         依赖倒置原则基于这样一个事实:相对于细节的多变性,抽象的东西要稳定的多。以抽象为基础搭建起来的架构比以细节为基础搭建起来的架构要稳定的多。在Java中,抽象指的是接口或者抽象类,细节就是具体的实现类,使用接口或者抽象类的目的是制定好规范和契约,而不去涉及任何具体的操作,把展现细节的任务交给他们的实现类去完成。

         依赖倒置原则的核心思想是面向接口编程,我们依旧用一个例子来说明面向接口编程比相对于面向实现编程好在什么地方。场景是这样的,母亲给孩子讲故事,只要给她一本书,她就可以照着书给孩子讲故事了。代码如下:

class Book{ 
 public String getContent(){ 
  return "很久很久以前有一个阿拉伯的故事……"; 
 } 
} 
class Mother{ 
 public void narrate(Book book){ 
  System.out.println("妈妈开始讲故事"); 
  System.out.println(book.getContent()); 
 } 
} 
public class Client{ 
 public static void main(String[] args){ 
  Mother mother = new Mother(); 
  mother.narrate(new Book()); 
 } 
} 

运行结果:

妈妈开始讲故事

很久很久以前有一个阿拉伯的故事……

        运行良好,假如有一天,需求变成这样:不是给书而是给一份报纸,让这位母亲讲一下报纸上的故事,报纸的代码如下:

class Newspaper{ 
 public String getContent(){ 
  return "林书豪38+7领导尼克斯击败湖人……"; 
 } 
} 

        这位母亲却办不到,因为她居然不会读报纸上的故事,这太荒唐了,只是将书换成报纸,居然必须要修改Mother才能读。假如以后需求换成杂志呢?换成网页呢?还要不断地修改Mother,这显然不是好的设计。原因就是Mother与Book之间的耦合性太高了,必须降低他们之间的耦合度才行。

我们引入一个抽象的接口IReader。读物,只要是带字的都属于读物:

interface IReader{ 
 public String getContent(); 
} 
Mother类与接口IReader发生依赖关系,而Book和Newspaper都属于读物的范畴,他们各自都去实现IReader接口,这样就符合依赖倒置原则了,代码修改为:
class Newspaper implements IReader { 
 public String getContent(){ 
  return "林书豪17+9助尼克斯击败老鹰……"; 
 } 
} 
class Book implements IReader{ 
 public String getContent(){ 
  return "很久很久以前有一个阿拉伯的故事……"; 
 } 
} 
class Mother{ 
 public void narrate(IReader reader){ 
  System.out.println("妈妈开始讲故事"); 
  System.out.println(reader.getContent()); 
 } 
} 
public class Client{ 
 public static void main(String[] args){ 
  Mother mother = new Mother(); 
  mother.narrate(new Book()); 
  mother.narrate(new Newspaper()); 
 } 
} 

运行结果:

妈妈开始讲故事
很久很久以前有一个阿拉伯的故事……
妈妈开始讲故事

林书豪17+9助尼克斯击败老鹰……   

这样修改后,无论以后怎样扩展Client类,都不需要再修改Mother类了。这只是一个简单的例子,实际情况中,代表高层模块的Mother类将负责完成主要的业务逻辑,一旦需要对它进行修改,引入错误的风险极大。所以遵循依赖倒置原则可以降低类之间的耦合性,提高系统的稳定性,降低修改程序造成的风险。   

采用依赖倒置原则给多人并行开发带来了极大的便利,比如上例中,原本Mother类与Book类直接耦合时,Mother类必须等Book类编码完成后才可以进行编码,因为Mother类依赖于Book类。修改后的程序则可以同时开工,互不影响,因为Mother与Book类一点关系也没有。参与协作开发的人越多、项目越庞大,采用依赖导致原则的意义就越重大。现在很流行的TDD开发模式就是依赖倒置原则最成功的应用。        

传递依赖关系有三种方式,以上的例子中使用的方法是接口传递,另外还有两种传递方式:构造方法传递和setter方法传递,相信用过Spring框架的,对依赖的传递方式一定不会陌生。
在实际编程中,我们一般需要做到如下3点:

  • 低层模块尽量都要有抽象类或接口,或者两者都有。
  • 变量的声明类型尽量是抽象类或接口。
  • 使用继承时遵循里氏替换原则。

        依赖倒置原则的核心就是要我们面向接口编程,理解了面向接口编程,也就理解了依赖倒置。


# java依赖倒转原则  # 依赖倒转原则  # Java设计模式之里氏替换原则精解  # Java设计模式之开闭原则精解  # Java设计模式之迪米特原则精解  # 详解java设计模式中的门面模式  # JAVA设计模式之单例模式详解  # java设计模式(实战)-责任链模式  # Java设计模式之职责链模式详解  # Java设计模式之观察者模式  # 详解Java设计模式中的装饰模式  # Java设计模式之依赖倒转原则精解  # 阿拉伯  # 很久  # 很久以前  # 有一个  # 抽象类  # 尼克斯  # 相对于  # 给孩子  # 的人  # 是一个  # 他们的  # 都有  # 也就  # 是这样  # 要有  # 两种  # 不去  # 给她  # 她就  # 要我 


相关栏目: 【 网站优化151355 】 【 网络推广146373 】 【 网络技术251813 】 【 AI营销90571


相关推荐: 如何用西部建站助手快速创建专业网站?  小米17系列还有一款新机?主打6.9英寸大直屏和旗舰级影像  网站制作大概多少钱一个,做一个平台网站大概多少钱?  成都品牌网站制作公司,成都营业执照年报网上怎么办理?  如何在局域网内绑定自建网站域名?  Laravel Blade组件怎么用_Laravel可复用视图组件的创建与使用  WordPress 子目录安装中正确处理脚本路径的完整指南  Laravel如何生成URL和重定向?(路由助手函数)  Laravel怎么配置.env环境变量_Laravel生产环境敏感数据保护与读取【方法】  Laravel如何集成Inertia.js与Vue/React?(安装配置)  js代码实现下拉菜单【推荐】  jQuery validate插件功能与用法详解  百度浏览器ai对话怎么关 百度浏览器ai聊天窗口隐藏  消息称 OpenAI 正研发的神秘硬件设备或为智能笔,富士康代工  Edge浏览器怎么启用睡眠标签页_节省电脑内存占用优化技巧  Python企业级消息系统教程_KafkaRabbitMQ高并发应用  Laravel如何理解并使用服务容器(Service Container)_Laravel依赖注入与容器绑定说明  郑州企业网站制作公司,郑州招聘网站有哪些?  如何用低价快速搭建高质量网站?  Win11搜索栏无法输入_解决Win11开始菜单搜索没反应问题【技巧】  JS中对数组元素进行增删改移的方法总结  php结合redis实现高并发下的抢购、秒杀功能的实例  ChatGPT常用指令模板大全 新手快速上手的万能Prompt合集  北京企业网站设计制作公司,北京铁路集团官方网站?  如何用花生壳三步快速搭建专属网站?  香港服务器网站搭建教程-电商部署、配置优化与安全稳定指南  图册素材网站设计制作软件,图册的导出方式有几种?  uc浏览器二维码扫描入口_uc浏览器扫码功能使用地址  Laravel Octane如何提升性能_使用Laravel Octane加速你的应用  Laravel项目结构怎么组织_大型Laravel应用的最佳目录结构实践  Laravel怎么为数据库表字段添加索引以优化查询  佛山企业网站制作公司有哪些,沟通100网上服务官网?  浅谈javascript alert和confirm的美化  Bootstrap整体框架之CSS12栅格系统  历史网站制作软件,华为如何找回被删除的网站?  Laravel如何实现一对一模型关联?(Eloquent示例)  HTML5段落标签p和br怎么选_文本排版常用标签对比【解答】  如何在云主机上快速搭建网站?  Laravel事件和监听器如何实现_Laravel Events & Listeners解耦应用的实战教程  如何快速搭建个人网站并优化SEO?  零服务器AI建站解决方案:快速部署与云端平台低成本实践  详解jQuery停止动画——stop()方法的使用  北京网站制作公司哪家好一点,北京租房网站有哪些?  Laravel怎么实现API接口鉴权_Laravel Sanctum令牌生成与请求验证【教程】  js实现点击每个li节点,都弹出其文本值及修改  网站设计制作书签怎么做,怎样将网页添加到书签/主页书签/桌面?  关于BootStrap modal 在IOS9中不能弹出的解决方法(IOS 9 bootstrap modal ios 9 noticework)  微博html5版本怎么弄发语音微博_语音录制入口及时长限制操作【教程】  如何在阿里云高效完成企业建站全流程?  悟空浏览器如何设置小说背景色_悟空浏览器背景色设置【方法】