[架构之路-183]-《软考-系统分析师》-13-系统设计 - 高内聚低耦合详解、图解以及技术手段

article/2025/9/24 21:59:52

目录

第1章  什么是高内聚低耦合

1.1 概念

1.2 目的

1.3 什么时候需要进行高内聚低耦合

1.4 什么系统需要关注高内聚、低耦合

第2章 分类

2.1 内聚的分类

2.2 耦合的分类

第3章  增加高内聚降低耦合度的方法

3.1 增加高内聚

3.2 降低耦合度


第1章  什么是高内聚低耦合

1.1 概念

高内聚低耦合,是软件工程中的概念,是判断软件设计好坏的标准,主要用于程序的面向对象的设计,主要看类的内聚性是否高,耦合度是否低。

内聚是从功能角度来度量模块内部的各子模块之间的相似性的程度,一个好的内聚模块应当恰好做一件事,它描述的是模块内的功能联系

耦合是从接口角度来度量模块与外界之间的相互连接、相互依赖、相互影响的一种度量,耦合强弱取决于模块间接口的复杂程度、进入或访问一个模块的点以及通过接口的数据。

1.2 目的

目的是增加程序模块的可重用性、增强程序的移植性,同时也有助于系统的有序性、

通常程序结构中各模块的内聚程度越高,模块间的耦合程度就越低

高内聚具备可靠性,可重用性,可读性等优点,模块设计推荐采用高内聚。

1.3 什么时候需要进行高内聚低耦合

在系统设计的概要设计阶段,对软件进行模块切分时,会使用到该概念。

需要注意的是,当你在做「分解」这个操作的时候,务必要关注每一次的「分解」是否满足一个最重要的条件:不同分支上的子问题,不能相互依赖,需要各自独立

因为一旦包含了依赖关系,子问题和父问题之间就失去了可以被「归并」的意义。

比如,一个「问题Z」被分解成了两个子问题,「子问题A」和「子问题B」。但是,解问题A依赖于问题B的答案,解问题B又依赖于问题A的答案。这不就等于没有分解吗?

架构是指可以预制和重构的软件框架结构。普遍指通过某种特定平台,而达到完成整体软件的功能。架构设计是指对软件、硬件、网络、运营、政策等软件设计中的需求和要素进行决策,主要包括体系结构设计和各个层的模块设计。

架构设计目标有个能够最大化的重用。

首先,要在架构的设计中灵活地使用各种共享的,特别是开源的框架技术因为共享的架构可以方便开发组分解问题,从而对项目中的功能模块分为需要内部解决和使用已有外部服务两类,避免了重复开发实现。

其次,尽量使用成熟的框架。由于服务器端软件系统的开发,涉及的知识、内容、要解决的技术问题很多,在某些方面使用第三方成熟的框架,相当于让别人帮助开发者完成了一些基础性的工作,此时开发者只需要集中精力完成系统业务逻辑的设计和实现。使软件系统实现可扩展性在技术上灵活地使用各种架构模式和代码设计模式,并且在使用代码设计模式的同时,使用其所提倡的面向接口编程,会对软件系统的可扩展性和可移植性的提高有所帮助。

希望能够设计出“高内聚、低耦合”的应用系统。这是架构设计最主要的目标,实现系统的高内聚、低耦合遵从以下原则:

  • 利用分层架构实现系统在纵向上的低藕合;利用开源框架进一步确保纵向分层的具体实现;
  • 按照功能划分子系统来实现横向上的低偶合;利用包结构确保横向上低耦合的具体实现。

1.4 什么系统需要关注高内聚、低耦合

只有一个大型的软件系统,需要对软件系统进行大量的模块切分时,才会需要到高内聚、低耦合设计思想。

一个小型的软件系统,是不需要考虑高内聚、低耦合。

第2章 分类

2.1 内聚的分类

内聚:故名思议,表示内部间聚集、关联的程度,那么高内聚就是指要高度的聚集和关联

内聚标志一个模块内各个元素彼此结合的紧密程度,它是信息隐蔽局部化概念的自然扩展。

内聚是从功能角度来度量模块内的联系,一个好的内聚模块应当恰好做一件事。它描述的是模块内的功能联系。

  • 偶然内聚:一个模块内的各处理元素之间没有任何逻辑上的联系,只是偶然地被凑到一起。这种模块也称为巧合内聚内聚程度最低
  • 逻辑内聚:这种模块把几种相关的功能组合在一起, 每次被调用时,由传送给模块参数来确定该模块应完成哪一种功能 。
  • 时间内聚:把需要同时执行的动作组合在一起形成的模块称为时间内聚模块。纯粹是因为执行时间相同,并没有功能/目标一致上的关系。
  • 先后过程内聚:构件或者操作的组合方式是,允许在调用前面的构件或操作之后,马上调用后面的构件或操作,即使两者之间没有数据进行传递。简单的说就是如果一个模块内的处理元素是相关的,而且必须以特定次序执行则称为过程内聚。例如某要完成登录的功能,前一个功能判断网络状态,后一个执行登录操作,显然是按照特定次序执行的。
  • 信息通信内聚:指模块内所有处理元素都在同一个数据结构上操作或所有处理功能都通过公用数据而发生关联(有时称之为信息内聚)。即指模块内各个组成部分都使用相同的数据结构或产生相同的数据结构
  • 逻辑顺序内聚:一个模块中各个处理元素和同一个功能密切相关,而且这些处理必须顺序执行,通常前一个处理元素的输出时后一个处理元素的输入。例如某要完成获取订单信息的功能,前一个功能获取用户信息,后一个执行计算均价操作,显然该模块内两部分紧密关联。顺序内聚的内聚度比较高,但缺点是不如功能内聚易于维护。
  • 功能内聚:模块内所有元素的各个组成部分全部都为完成同一个功能或目标而存在,共同完成一个单一的功能,模块已不可再分。即模块仅包括为完成某个功能所必须的所有成分,这些成分紧密联系、缺一不可,不多不少,凝聚力强!

2.2 耦合的分类

耦合:是对模块间关联程度的度量。

耦合的强弱取决与模块间接口的复杂性、调用模块的方式以及通过界面传送数据的多少。

模块间的耦合度是指模块之间的依赖关系,包括:

  • 控制关系
  • 调用关系
  • 数据传递关系。

模块间联系越多,其耦合性越强,同时表明其独立性越差。

降低模块间的耦合度能减少模块间的影响,防止对某一模块修改所引起的“牵一发动全身”的水波效应,保证系统设计顺利进行。

耦合度就是某模块(类)与其它模块(类)之间的关联、感知和依赖的程度,是衡量代码独立性的一个指标。

  • 非直接耦合:两个模块之间没有直接关系,它们之间的联系完全是通过主模块控制调用实现的耦合度最弱,模块独立性最强。
  • 函数调用/发送消息:值数据耦合:调用模块被调用模块之间只传递简单的数据项参数。相当于高级语言中的值传递
  • 函数调用/发送消息:标记/地址耦合:调用模块和被调用模块之间传递数据结构而不是简单数据,同时也称作特征耦合。模块间传递的不是简单变量值,而是像高级语言中的数据名、记录名和文件名等数据的标记,这些名字即为标记,其实传递的是地址
  • 备注:数据耦合相对于控制耦合,是松耦合,因为万事万物,总会有些信息联系的。
  • 函数调用/发送消息:控制耦合:模块之间传递的不是数据信息,而是控制信息,例如标志、开关量等,一个模块控制了另一个模块的功能。
  • 共享资源:外部变量耦合、外部耦合:一组模块都访问同一全局简单变量,而且不通过参数表传递该全局变量的信息,则称之为外部耦合。
  • 共享资源:公共数据耦合:一组模块都访问同一个全局数据结构,则称之为公共耦合。公共数据环境可以是全局数据结构、共享的通信区、内存的公共覆盖区等。如果模块只是向公共数据环境输入数据,或是只从公共数据环境取出数据,这属于比较松散的公共耦合;如果模块既向公共数据环境输入数据又从公共数据环境取出数据,这属于较紧密的公共耦合。
  • 内容耦合:是指如果一个模块与另一个模块的内部属性有关,不经调用直接使用另一个模块的程序代码或内部数据,那么这两个模块之间就存在内容耦合。这种耦合表明一个模块与另一个模块的内部数据或程序代码有关,当一个模块的程序代码被修改或内部数据出错,必然引起另一个模块出错。而对后一模块的出错是很难查出原因的,这样给模块的修改、维护带来极大困难。内容耦合的耦合度最大,为“病态耦合”,在设计时,应避免这种耦合。在高级语言中,实际上已经规避了该类型的耦合,只有在汇编语言中,一个模块、函数才能直接访问另一个函数内部的变量和内存空间!!!

备注0:

共享资源虽然是高耦合,但并非一无是处,当两个模块之间需要高性能数据交换时,共享内存(共享数据)是一个非常高效的手段!!!

备注1:不存在完全孤立的模块

没有与外界完全没有耦合的模块,与外界完全没有耦合的模块是信息孤岛,是没有存在的意义和价值的!!!!

备注2:公共耦合会引起以下问题

1. 无法控制各个模块对公共数据的存取,严重影响了软件模块的可靠性和适应性

2. 使软件的可维护性变差。若一个模块修改了公共数据,则会影响相关模块。

3. 降低了软件的可理解性。不容易清楚知道哪些数据被哪些模块所共享,排错困难。

一般地,仅当模块间共享的数据很多且通过参数传递很不方便时,才使用公共耦合。

备注3:内容耦合

一个模块直接访问另一模块的内容,则称这两个模块为内容耦合。

若在程序中出现下列情况之一,则说明两个模块之间发生了内容耦合:

1. 一个模块直接访问另一个模块的内部数据。

2. 一个模块不通过正常入口而直接转入到另一个模块的内部。

3. 两个模块有一部分代码重叠(该部分代码具有一定的独立功能)。

4. 一个模块有多个入口。

内容耦合可能在汇编语言中出现。大多数高级语言都已设计成不允许出现内容耦合。这种耦合的耦合性最强,模块独立性最弱。

第3章  增加高内聚降低耦合度的方法

3.1 增加高内聚

  • 1、模块只对外暴露最小限度的接口,暂时不需要的接口,不需要暴露,形成最低的依赖关系。
  • 2、只要对外接口不变,模块内部的修改,就不得影响其他模块。
  • 3、删除一个模块,应当只影响有依赖关系的其他模块,而不应该影响其他无关部分。

3.2 降低耦合度

1、少使用类的继承,多用接口隐藏实现的细节。

Java面向对象编程引入接口除了支持多态外, 隐藏实现细节也是其中一个目的。

继承能很好的利用现有内的结构化定义,但也会导致两个类深度耦合。

2、模块的功能化分尽可能的单一

道理也很简单,功能单一的模块供其它模块调用的机会就少。

(其实这是高内聚的一种说法,高内聚低耦合一般同时出现)。

3、遵循一个定义只在一个地方出现。

4、少使用全局变量

5、类属性和方法的声明少用public,多用private关键字。

public的本质就是类的接口,接口越多,对外的依赖性越大。public属性越少,与外界的接口越少。

6、多用设计模式,比如采用MVC的设计模式就可以降低界面与业务逻辑的耦合度。

设计模式本身,就是遵循高内聚、低耦合的原则设计的模式!!!

7、尽量不用“硬编码”的方式写程序,同时也尽量避免直接用SQL语句操作数据库。

摆脱业务代码与数据库的强耦合。

8、最后当然就是避免直接操作或调用其它模块或类的数据(内容耦合);

如果模块间必须存在耦合,原则上尽量使用数据耦合少用控制耦合限制公共耦合的范围避免使用内容耦合


http://chatgpt.dhexx.cn/article/kej8Pfkp.shtml

相关文章

java 高内聚低耦合_关于高内聚低耦合概念的理解

概念 高内聚低耦合,是软件工程中的概念,是判断软件设计好坏的标准,主要用于程序的面向对象的设计,主要看类的内聚性是否高,耦合度是否低。 目的 使程序模块的可重用性、移植性大大增强。 通常程序结构中各模块的内聚程…

lea 指令的小解

LEA指令的功能是将源操作数、即存储单元的有效地址(偏移地址)传送到目的操作数。 官方8086的白皮书,我的英语水平有限,只找到如下内容; ^offset和 Lea 伪指令 ; offset 此为一个伪指令,它不会有…

X86 LEA指令详解

1.SDM指令功能描述(LEA) LEA 总体描述: 从第二个操作数(源操作数)计算有效地址,并将结果存入第一个操作数(目的操作数)。源操作数是指定了一种访存操作的内存地址,目的操作数为一个通用寄存器…

css3围绕旋转动画js特效

下载地址 css3围绕旋转动画特效,两种旋转方式,自转和公转,css3按照固定的轨迹行驶。 dd:

HTML5 - 用CSS3动画制作场景切换效果(移动,旋转,淡入淡出等)

两个场景(即两个div视图)切换的时候,如果想添加个过渡动画,除了可以使用js来实现,还可以通过CSS3的animation属性来实现。 (注意:Internet Explorer 9 以及更早的版本不支持 animation 属性。&…

CSS3 元素转圈动画 (元素旋转动画)

CSS3 元素转圈动画 &#xff08;元素旋转动画&#xff09; <!DOCTYPE html> <html> <head> <style> div { width:100px; height:100px; background:red; position:relative; animation:mymove 5s infinite; -webkit-animation:mymove 5s infinite; /*…

css3 animation动画360旋转。旋转效果用transform:rotate过渡。

旋转&#xff1a; <div class"lds-hourglass"></div> .lds-hourglass {display:inline-block;position:relative;width:72px;height:72px } .lds-hourglass:after {content:" ";display:block;border-radius:50%;width:0;height:0;margin:6p…

css的rotate3d实现炫酷的圆环转动动画

1.实现效果 2.实现原理 2.1 rotate3d rotate3d&#xff1a;rotate3d() CSS 函数定义一个变换&#xff0c;它将元素围绕固定轴移动而不使其变形。运动量由指定的角度定义; 如果为正&#xff0c;运动将为顺时针&#xff0c;如果为负&#xff0c;则为逆时针。 语法&#xff1a; …

CSS3实现3d图片旋转动画效果

1.实现原理 1.首先所有的图片的容器position:absolute&#xff0c;叠加在一起&#xff0c;然后一次设置rotateY分别为&#xff08;36*i&#xff09;deg &#xff0c;i取0到10 &#xff1b;所有图片会相交成一个类似花的形状 然后为每个图片的容器设置translateZ&#xff0c;所有…

CSS3实现立体旋转动画效果

问题描述 一个3D立体图形在网页中呈现旋转效果,看起来十分炫酷,这个过程的实现其实并不困难,只需要利用CSS3动画效果,并对属性设置合适的参数之后,效果就能实现了。 在解决问题之前,需要先了解CSS3的动画效果是什么,其相关属性和参数有什么含义,这个过程就请查看CSS3动…

css3d动画:平移、旋转、缩放

1、前言&#xff1a; 3d比2d多了一个z轴&#xff0c;这个z轴是垂直我们屏幕的方向&#xff0c;指向我们人眼的是正轴&#xff0c;远离人眼的是负轴&#xff0c;图示如下&#xff1a; 2、景深设置和3d平移 当我们在z轴上向正轴方向上移动物体&#xff0c;也就是向我们眼前移动…

使用css3属性做一个循环旋转的动画

做这个动画是使用css3中的animation 和 @-webkit-keyframes 组合使用来完成 //这是html部分代码 <div class="home"><imgalt="logo"src="http://b-ssl.duitang.com/uploads/item/201206/26/20120626190359_MjB3s.jpeg"id="images…

基于HTML5+CSS3的图片旋转、无限滚动、文字跳动特效

本文分享几种基于HTML5CSS3实现的一些动画特效&#xff1a;图片旋转、无限滚动、文字跳动;实现起来均比较容易&#xff0c;动手来试试! 一、图片旋转 效果图如下&#xff1a; 这个效果实现起来其实并不困难。代码清单如下: <style type"text/css"> #liu{ wid…

内存碎片产生原因及终极解决办法

内存碎片通常分为内部碎片和外部碎片&#xff1a; 1. 内部碎片是由于采用固定大小的内存分区&#xff0c;当一个进程不能完全使用分给它的固定内存区域时就产生了内部碎片&#xff0c;通常内部碎片难以完全避免&#xff1b; 2. 外部碎片是由于某些未分配的连续内存区域太小&…

6、Redis内存碎片优化

1、Redis内存碎片是如何形成的 一是内存分配器的分配策略&#xff0c;内存分配器一般是按固定大小来分配内存&#xff0c;而不是按实际使用大小来分配。例如8字节、16字节、32字节...&#xff0c;2KB&#xff0c;4KB&#xff0c;8KB等&#xff0c;按程序申请时最接近某个固定值…

内存碎片处理技术

内存碎片处理技术 内存碎片是一个很棘手的问题。如何分配内存决定着内存碎片是否会、何时会、如何会成为一个问题。 即使在系统中事实上仍然有许多空闲内存时&#xff0c;内存碎片还会最终导致出现内存用完的 情况。一个不断产生内存碎片的系统&#xff0c;不管产生的内存碎片…

内存碎片产生原因及处理

内存碎片是一个很棘手的问题。如何分配内存决定着内存碎片是否会、何时会、如何会成为一个问题。  即使在系统中事实上仍然有许多空闲内存时&#xff0c;内存碎片还会最终导致出现内存用完的情况。一个不断产生内存碎片的系统&#xff0c;不管产生的内存碎片多么小&#xff0…

什么是Redis内存碎片率?碎片如何清理?

Redis服务器内存不够用了&#xff0c;这时可以看看是否是内存碎片的原因&#xff0c;不一定只能靠内存扩容来处理。 一、碎片率的计算 我们登陆到Redis服务器上&#xff0c;执行以下命令&#xff1a; info memory我们会看到如下的信息&#xff1a; 其中mem_fragmentation_r…

Redis 内存碎片详解

什么是内存碎片? 你可以将内存碎片简单地理解为那些不可用的空闲内存。 举个例子&#xff1a;操作系统为你分配了 32 字节的连续内存空间&#xff0c;而你存储数据实际只需要使用 24 字节内存空间&#xff0c;那这多余出来的 8 字节内存空间如果后续没办法再被分配存储其他数…

redis之内存碎片问题如何解决

写在前面 本文一起看下如何处理Redis的内存碎片问题。 1&#xff1a;什么是内存碎片 当操作系统内存充足&#xff0c;但是却无法申请连续N字节内存空间时&#xff0c;这些剩余的内存空间就叫内存碎片。 2&#xff1a;内存碎片是如何形成的 内存碎片的形成有两方面的因素&a…