解耦:哪些方法可以用来解耦代码

目录

1.引用

2.为何解耦如此重要

3.如何判断代码是否需要解耦

4.如何给代码解耦

5.思考题


1.引用

        前面我们曾经讲到,重构可以分为大型重构和小型重构。小型重构的主要目的是提高代码的可读性,大型重构的主要目的是解耦。本节讲解如何对代码进行解耦。

2.为何解耦如此重要

        在软件的设计与开发过程中,我们需要关注代码的复杂度问题。复杂的代码经常有可读性、可维护性方面的问题,那么,如何控制代码的复杂度呢?其实,控制代码的复杂度的手段有很多,效果显著的应该是解耦,因为解耦可以使代码高内聚、低耦合。利用解耦的方式对代码进行重构可以有效控制代码的复杂度。

        实际上,”高内聚、低耦合”是一种通用的设计思想,它不仅可以指导细粒度的类之向关系的设计,还能指导粗粒度的系统、架构、模块的设计。相比代码规范,它能够在更高层次上提高代码的可读性和可维护性。

        无论是阅读代码还是修改代码,“高内聚、低耦合”特性可以让我们聚焦在某一模块或类上,不需要过多了解其他模块或类的代码,从而降低阅读代码和修改代码的难度。因为依赖关系简单,耦合度低,所以修改代码时不会牵一发而动全身,代码改动集中,引入bug的风险降低。

        代码“高内聚、低耦合”意味着代码的结构清晰,分层和模块化合理,依赖关系简单,模块或类之间的耦合度低。对于“高内聚、低耦合”的代码,即使某个类或模块内部的设计不合理,代码质量不算高,影响范围也是有限的。我们可以聚焦这个模块或类并进行小型重构相比代码结构的调整,这种改动集中的小型重构的难度大幅降低。

3.如何判断代码是否需要解耦

        如果修改一段功能代码时出现“牵一发而动全身”的情况,那么说明这个项目的代码耦合度过高,需要对其进行解耦。除此之外,我们还有一个直观的衡量方式,就是先把项目代码中的模块之间、类之间的依赖关系画出来,再根据依赖关系图的复杂度来判断项目代码是否需要解耦。如果模块之间、类之间的依赖关系复杂、混乱,那么说明代码结构存在问题,此时,我们可以通过解耦让依赖关系变得简单、清晰。

4.如何给代码解耦

        接下来,我们探讨一下如何给代码解耦。

        1.通过封装与抽象来解耦

        封装和抽象可以应用在多种代码设计场景中,如系统、块、类库、组件、接口和类等的设计。封装和抽象可以有效地隐藏实现的复杂性,隔离实现的易变性,给上层模块提供稳定目易用的接口。

        例如,UNIX系统提供的文件操作函数open()使用简单,但其底层实现复杂,涉及权限控制、并发控制和物理存储等。我们通过将open()封装为一个抽象的函数,能够有效控制代码复杂性的蔓延,将代码复杂性封装在局部代码中。除此之外,因为open()函数基于抽象而非具体实现来定义,所以我们在改动open()函数的底层实现时,并不需要改动依赖它的上层代码。

        2.通过引入中间层来解耦

        中间层能够简化模块之间或类之间的依赖关系。图5-1是引入中间层前后的依赖关系对比图。在引入数据存储中间层之前,A、B和C模块都要依赖内存一级缓存、Redis 二级缓存和DB 持久化存储3个模块。在引入数据存储中间层之后,A、B和C模块只需要依赖数据存中间层模块。从图5-1可以看出,中间层的引入简化了模块之间的依赖关系,让代码结构更加清晰。

        在进行重构时,中间层可以起到过渡作用,实现开发和重构同步进行,且不互相干扰。例如,某个接口的设计有问题,我们需要修改它的定义,于是,所有调用这个接口的代码都要做相应改动。如果新开发的代码也使用这个接口,那么开发与重构之间会产生冲突。为了使重构“小步快跑”,我们可以通过以下4个阶段完成对接口的修改。

        1)第一阶段:引入一个中间层,利用中间层“包裹”旧接口,提供新接口。

        2)第二阶段:新开发的代码依赖中间层提供的新接口。

        3)第三阶段:将依赖旧接口的代码改为调用新接口。

        4)第四阶段:确保所有代码中都调用新接口之后,删除旧接口。通过引入中间层,我们可以分阶段完成重构。由于每个阶段的开发工作量都不会很大,可以在短时间内完成,因此重构与开发发生冲突的概率变小了。

        3.通过模块化、分层来解耦

        模块化是构建复杂系统的常用手段。模块化还广泛用于建筑、机械制造等行业。对于UNIX这样复杂的系统,我们很难掌控其所有实现细节。之所以人们能够开发出UNIX这样复杂的系统,并且能够对其进行维护,主要原因是将该系统划分成了多个独立模块,如进程调度、进程通信、内存管理、虚拟文件系统和网络接口等模块。模块之间通过接口通信,模块之间的耦合度很小,每个小型团队负责一个独立的高内聚模块的开发,最终,将各个模块组合构成一个复杂的系统。

        实际上,模块化思想在SOA(Service-Oriented Architecture,面向服务的架构)、微服务类库,以及类和函数的设计等方面都有所体现。模块化的本质是“分而治之”。

        我们将目光聚焦到代码层面。在开发代码时,我们要有模块化意识,将每个模块都当作个独立的类库来开发,只提供封装了内部实现细节的接口给其他模块使用,这样可以降低模块之间的耦合度。

        除模块化以外,分层也是构建复杂系统的常用手段。例如,UNIX系统就是基于分层思想之间的耦合度。开发的,它大致分为3层:内核层、系统调用层和应用层。每一层都封装了实现细节,并且暴露抽象的接口供上层使用。而且,任意一层部可以被重新实现,不会影响其他层的代码。面对复杂系统的开发,我们要善于应用分展技术,尽量将容易复用、与具体业务关系不大的代码下沉到下层,将容易变动、与具体业务强相关的代码移到上层。

        4.利用经典的代码设计思想和设计原则来解耦

        我们总结一下可以用来解耦的代码设计原则和设计思想。

        (1)单一职责原则

        内聚性和耦合性二者并非相互独立。高内聚使得代码低耦合,而实现高内聚的重要指导则是单一职责原则。如果模块或类的职责单一,那么依赖它们的类和它们依赖的类较少,代码的耦合度也就降低了。

        (2)基于接口而非实现编程

        如果我们利用“基于接口而非实现编程”思想来编程,那么,在有依赖关系的两个模块或类之间,一个模块或类的改动不会影响另一个模块或类。这就相当于将一种强依赖关系(强合)解耦为了弱依赖关系(弱耦合)。

        (3) 依赖注入

        与“基于接口而非实现编程”类似,依赖注入也能将模块或类之间的强耦合变为弱耦合尽管依赖注入无法将本应该有依赖关系的两个类解耦为没有依赖关系,但可以使二者的耦合关系不再像原来那么紧密,方便将某个类锁依赖的类替换为其他类。

        (4)多用组合,少用继承

        继承是一种强依赖关系,父类与子类高度耦合,且这种耦合关系非常脆弱,父类的每一次改动都会影响其所有子类。组合是一种弱依赖关系。对于复杂的继承关系,我们可以利用组合替换继承,以达到解耦的目的。

        (5) LoD

        LoD 的定义描述是:不应该存在直接依赖关系的类之间不要有依赖,有依赖关系的类之间量只依赖必要的接口。从LoD的定义描述中可以看出,使用LoD的目的就是实现代码的低耦合。除上述设计思想和设计原则以外,大部分设计模式也能起到解耦的效果,关于这一部分内容,在我们之前都有讲过。

5.思考题

        实际上,在平时的开发中,解耦到处可见,例如,Spring中的AOP能实现业务代码与非业务代码的解耦,IoC 能实现对象的创建和使用的解耦,除此之外,读者还能想到哪些解耦场景?

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

webgl three 模型操作

模型位置操作是改变相关属性 平移改变位置 缩放改变缩放值 旋转改变角度 webgl中是4x4矩阵 前三排是原始状态 第四排是改变值 .position 位置 .scale 缩放 .rotation 绕轴旋转 .quaternion 绕axis旋转 其实和svg差不多 缩放 缩放是操作对角线 // Matrix4x4 …

大语言模型的工程技巧(三)——分布式计算

相关说明 这篇文章的大部分内容参考自我的新书《解构大语言模型:从线性回归到通用人工智能》,欢迎有兴趣的读者多多支持。 本文将讨论如何利用多台机器进行神经网络的分布式训练。利用多台机器来加速大语言模型的训练,是其获得成功的重要原…

操作系统总结3

目录 2.3.3 进程互斥的硬件的实现方法 (1)知识总览 (2)中断的屏蔽方法 (3)TestAndSet指令 (4)Swap指令 总结 2.3.4 信号量机制 (1)知识总览 &#x…

《web应用设计》第八次作业

我的小组长是姚若希,我们组课程设计的题目是:学生管理系统 ,我认领的功能模块是:课程管理 2.查询并分页

MM模块五(采购订单)

创建采购订单:ME21N 参考报价单创建采购订单: 哪个采购部门的哪个采购员为哪个公司代码下的采购订单 点击保存 注:采购订单的数量不能大于采购申请的数量

选择优化求解器的关键因素:以MindOpt为例

选择一款适合自己业务需求的求解器我们一般需要考量什么呢?可求解的问题类型?问题规模?本文将介绍一些需要考虑的重要因素,并且介绍阿里达摩院MindOpt优化求解器在这些因素下的表现。 ⬇️⬇️⬇️⬇️⬇️⬇️⬇️⬇️⬇️⬇️⬇…

【Linux】Centos7安装MySql

【Linux】Centos7安装MySql 下载 MySQL 官网下载 MySQL,用的是 CentOS7.9 所以选择红帽系统 https://downloads.mysql.com/archives/community/ 安装 使用rz命令上传 MySQL tar # 创建 MySQL 解压目录 [rootlocalhost /]# mkdir /usr/local/mysql# 解压 [rootloca…

【漏洞复现】懒人网址导航页 search.html SQL注入漏洞

0x01 产品简介 赖人网址导航系统是一种智能化的网址导航平台,旨在帮助用户快速找到所需的网址和资源。该系统提供了智能化的网址搜索和推荐功能,能够根据用户的搜索习惯和偏好推荐相关的网址和资源。同时,系统还提供了网址分类、网址收藏和网…

VScode SSH连接远程服务器报错

一、报错 通过VScode SSH插件远程连接服务器,输入密码后没有连接成功,一直跳出输入密码界面,在输出界面里,一直是Waiting for server log或者是显示Cannot not find minimist 二、处理 🐱: 这个时候应该…

数据可视化技术头歌测试合集

努力是为了不平庸~ 学习的最大理由是想摆脱平庸,早一天就多一份人生的精彩;迟一天就多一天平庸的困扰 目录 时间趋势可视化-柱形图 第1关:“大胃王”比赛数据柱形图绘制——绘制柱形图的基本步骤 任务描述 相关知识 观察和处理数据 绘…

JLINK使用及问题

一、安装:安装过程选择好文件夹,过程中一直默认就行,如果之前安装的有其他版本,在最后一步需要多注意一下,或者不注意,直接默认选中的就行。 二、使用: 在官网下载的J-Link V7.96i&#xff0c…

视频码流分析工具

一、VQ Analyzer 在线使用说明: https://vicuesoft.com/vq-analyzer/userguide/ ref: Video Analyzer and Streaming Tester Software – VQ Analyzer HEVC 分析工具 - 懒人李冰 推荐一个开源且跨平台的免费码流分析软件YUView - 知乎

【大模型部署】在C# Winform中使用文心一言ERNIE-3.5 4K 聊天模型

【大模型部署】在C# Winform中使用文心一言ERNIE-3.5 4K 聊天模型 前言 今天来写一个简单的ernie-c#的例子,主要参考了百度智能云的例子,然后自己改了改,学习了ERNIE模型的鉴权方式,数据流的格式和简单的数据解析,实…

【漏洞复现】智慧校园(安校易)管理系统 FileUpProductupdate.aspx 任意文件上传漏洞

0x01 产品简介 "安校易"是银达云创公司基于多年教育市场信息化建设经验沉淀,经过充分的客户需求调研,并依据国家"十三五”"教育信息化建设规范而推出的综合互联网教育信息化解决方案。“安校易“以物联网技术为基础,以学…

民国漫画杂志《时代漫画》第19期.PDF

时代漫画19.PDF: https://url03.ctfile.com/f/1779803-1248634637-c04860?p9586 (访问密码: 9586) 《时代漫画》的杂志在1934年诞生了,截止1937年6月战争来临被迫停刊共发行了39期。 ps: 资源来源网络!

oracle怎么处理json格式

向数据库导入json相关jar包 loadjava -r -f -u bsuser/XXXX192.168.10.31/bsorcl json.jar 要删除的话,删除指定jar dropjava -u bsuser/XXXX192.168.10.31/bsorcl json.jar select * from user_java_classes 然后我们就可以取到json串中任意节点的值

UWB论文:Introduction to Impulse Radio UWB Seamless Access Systems(2):脉冲;超宽带;测距;定位

3) 测距/接收器 像全球定位系统(GPS)这样的系统依赖于单向测距One Way Ranging(OWR),其中多个卫星(代表固定节点,称为锚点anchors)定期传输同步的无线电数据包集合,这允许…

stream( ).collect ( Collectors.groupingBy ( ) ) 的用法

文章目录 第一种解释1、基本用法2、指定值收集器3、多级分组4、常见应用场景和用处 第二种解释1、基本语法2、示例3、更复杂的用法 第一种解释 Collectors.groupingBy 是 Java 8 引入的 Stream API 中的一个收集器(Collector),它用于将流&am…

AI - 各类AI针对Excel分析对比

一个水果销量表,Excel包含多个年份sheet,需要提取某个品种的水果每年的销量,看看几个AI的分析结果吧 1、文心一言3.5(不支持Excel) 不支持上传Excel文件 2、 通义千问2.5(完成★) 顺利完成…

web前端学习笔记11

11. CSS3高级特效 11.1 CSS3变形 CSS3变形是一些效果的集合, 如平移、旋转、缩放、倾斜效果 每个效果都可以称为变形(transform),它们可以分别操控元素发生平移、旋转、缩放、倾斜等变化 语法 transform:[transform-function] ; /* 设置变形函数,可以是一个,也可以是多…