如何利用迪米特法则实现“高内聚、低耦合”?

作者:JavaEdge

1.何为“高内聚、低耦合”?

“高内聚、低耦合”能有效地提高代码可读性、可维护性,缩小功能改动导致的代码改动范围。很多设计原则也都以实现代码“高内聚、低耦合”为目的,比如:

单一职责原则面向接口,而非面向实现来编程“高内聚、低耦合”是个通用设计思想,可指导:

不同粒度代码的设计与开发如系统、模块、类,甚至函数

不同开发场景如微服务、框架、组件、类库等本文主要围绕以“类”作为该设计思想的应用对象。

“高内聚”,指导类本身的设计“低耦合”,指导类与类之间依赖关系的设计二者并非完全独立,高内聚有助于低耦合,低耦合又需要高内聚的支持。

1.1 高内聚

相近的功能,应放到同一个类不相近的功能,不要放到同一个类相近的功能往往会被同时修改,放到同一类中,修改会比较集中,代码易维护。单一职责原则就是实现代码高内聚非常有效的设计原则。

1.2 低耦合

在代码中,类与类之间的依赖关系简单清晰。

即使两个类有依赖关系,一个类的代码改动不会或很少会导致依赖类的代码改动。依赖注入、接口隔离、面向接口编程及迪米特法则都是为实现低耦合。

1.3 “内聚”和“耦合”的关系

左边代码结构是“高内聚、低耦合”;右边“低内聚、紧耦合”:

左边的代码设计:

类的粒度较小,每个类的职责都比较单一。相近功能都放到了一个类,不相近功能分割到多个类。这样类更加独立,代码内聚性更好。

因为职责单一,所以每个类被依赖的类就会比较少,代码低耦合。一个类的修改,只会影响到一个依赖类的代码改动。只需要测试这一个依赖类是否还能正常工作即可。

右边代码设计:

类粒度比较大,低内聚,功能大而全,不相近的功能放到了一个类中。导致很多其他类都依赖该类。修改这个类的某功能代码时,会影响依赖它的多个类。我们需要测试这三个依赖类,是否还能正常工作,“牵一发而动全身”。

2.迪米特法则

Law of Demeter,LOD,从名字看不出这是个啥。它还有另外一个名字:最小知识原则。

The Least Knowledge Principle:Each unit should have only limited knowledge about other units: only units “closely” related to the current unit. Or: Each unit should only talk to its friends; Don’t talk to strangers.

每个模块都只应该了解那些与它关系密切的模块(units: only units “closely” related to the current unit)的有限知识(knowledge)。或者说,每个模块只和自己的朋友“说话”(talk),不和陌生人“说话”(tal‍k)。

结合实际,定义描述中的“模块”替换成“类”:

不该有直接依赖关系的类之间,不要有依赖有依赖关系的类之间,尽量只依赖必要的接口(“有限知识”)所以,迪米特法则其实包含两部分,下面用两个案例分别解读。

3.案例

3.1 不该有直接依赖关系的类之间,不要有依赖

简化的搜索引擎爬取网页,包含如下主要类:

NetworkTransporter,负责底层网络通信,根据请求获取数据:

HtmlDownloader,通过URL获取网页:

Document,表示网页文档,后续的网页内容抽取、分词、索引都是以此为处理对象:

如何重构?

这段代码虽然“能用”,但不够“好用”。

NetworkTransporter,作为一个底层网络通信类,应该尽可能通用,而不是只能下载HTML。所以,不应该直接依赖太具体的发送对象HtmlRequest,其设计违背迪米特法则,依赖了不该有直接依赖关系的HtmlRequest类。

假如你现在要去买东西,你肯定不会直接把钱包给收银员,让收银员自己从里面拿钱,而是你从钱包里把钱拿出来交给收银员。HtmlRequest对象相当于钱包,HtmlRequest里的address和content对象就相当于钱。

应将address和content交给NetworkTransporter,而非直接把HtmlRequest交给NetworkTransporter:

Document的问题:

构造器中的downloader.downloadHtml()逻辑复杂,耗时长,不应放到构造器,影响代码可测试性HtmlDownloader对象在构造器通过new来创建,违反面向接口编程,也影响代码可测试性业务上说,Document网页文档没必要依赖HtmlDownloader类,违背迪米特法则问题虽多,但修改简单:

3.2 有依赖关系的类之间,尽量只依赖必要的接口。

Serialization类负责对象的序列化和反序列化:

单看类的设计,没问题。但若放到特定应用场景,假设项目中的有些类只用到序列化操作,而另一些类只用到反序列化。那么,基于 有依赖关系的类之间,尽量只依赖必要的接口:

只用到序列化操作的那部分类不应依赖反序列化接口只用到反序列化操作的那部分类不应依赖序列化接口据此,应将Serialization类

方案一:拆分为两个更小粒度的类

只负责序列化(Serializer类)只负责反序列化(Deserializer类)

尽管拆分后的代码更能满足迪米特法则,但却违背高内聚。高内聚要求相近功能放到同一类中,方便功能修改时,修改的地方不太散乱。

针对本案例,若业务要求修改序列化实现方式,从JSON换成XML,则反序列化实现逻辑也要一起改。未拆分前,只需修改一个类,拆分后,却要修改两个类!

既不想违背高内聚,也不想违背迪米特法则,怎么办?

方案二:引入两个接口

尽管还是要往Demo1的构造器传入包含序列化和反序列化的Serialization实现类,但依赖的Serializable接口只包含序列化操作,Demo1无法使用Serialization类中的反序列化接口,对反序列化操作无感知,符合迪米特法则的“依赖有限接口”。

也体现了“面向接口编程”,结合迪米特法则,可总结出:“基于最小接口,而非最大实现来编程”。

多想一点点

本案例的重构方案,整个类只包含序列化、反序列化俩操作,只用到序列化操作的使用者,即便能够感知到仅有的一个反序列化方法,问题也不大。为满足迪米特法则,将一个简单的类,拆出两个接口,是过度设计吗?设计原则本身无对错,只有能否用对之说。不要为了用设计原则而用,应该具体问题具体分析。

Serialization类只包含两个操作,确实没啥必要拆成俩接口。但若对Serialization类添加更多功能,实现更多更好用的序列化、反序列化方法,重新考虑该问题:

这种场景下,方案二设计更好。因为本案例的应用场景,大部分代码只用到序列化功能,这些用户无需了解反序列化,而修改后的Serialization类,反序列化的“知识”,从一个方法变成三个。一旦任一反序列化操作有代码改动,都需要检查、测试所有依赖Serialization类的代码是否还能正常工作。

为减少耦合和降低测试的工作量,应按迪米特法则,隔离反序列化和序列化的功能。

4.总结

4.1 高内聚、低耦合

能有效提高代码的可读性和可维护性,缩小功能改动导致的代码改动范围:

高内聚,指导类本身的设计就是指相近的功能应该放到同一个类中,不相近的功能不要放到同一类中。相近的功能往往会被同时修改,放到同一个类中,修改会比较集中

低耦合,指导类与类之间依赖关系的设计在代码中,类与类之间的依赖关系简单清晰。即使两个类有依赖关系,一个类的代码改动也不会或者很少导致依赖类的代码改动。

4.2 迪米特法则

不该有直接依赖关系的类之间,不要有依赖;有依赖关系的类之间,尽量只依赖必要的接口。迪米特法则希望减少类之间的耦合,让类越独立越好。每个类都应该少了解系统的其他部分。一旦发生变化,需要了解这一变化的类就会比较少。

参考:

[1]. 《重构》

[2]. 《代码设计之丑》

来源: JavaEdge

原文地址:https://www.toutiao.com/article/7080826821444469285/

版权声明:本文内容由互联网用户自发贡献,该文观点与技术仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 dio@foxmail.com 举报,一经查实,本站将立刻删除。

相关推荐


所以很多人都会选择将这些生活琐事来交给智能化产品,在众多产品中,扫拖机器人所给我们带来的便利性最强,扫地、拖地全都一气呵成,不需要人为过多干预,不过目前大多数扫拖机器人对于双手解放得不彻底。而石头作为
“昨天的经历都将成为明天的力量。” 将绝大部分精力都奉献给璃月港的刻晴,可以说是《原神》中的高人气角色了,虽然是常驻角色,并且对于普通玩家来说……刻师傅别刮了……不过作为开服就存在的角色,还有独特的剑法
最近,全球知名的通信产业盛会MWC 2024(2024世界移动通信大会)正式召开,其中,联发科以“连接AI宇宙”(Connecting the AI-verse)为主题,为大众展示出一系列在AI和移动通信技术等领域的最新突破,吸引了大量行业
今年上半年有很多值得关注的机型,其中华为最新的影像旗舰华为P70 Art也自然受到了业界不少的关注目光,目前关于这款机型的轮廓图已经在网上曝光。
目前,2024世界移动通信大会(MWC)正在西班牙巴塞罗那举行,值得一提的是,此次大会参展中国厂商非常多,包括华为、中兴、小米、荣耀等等多家厂商均在其列。
就在去年,真我推出了11 Pro+,用一个2亿像素传感器和zoom变焦功能,开启了中端手机影像的长焦大战,而后友商才姗姗来迟的跟进了2亿像素传感器。
【手机之家新闻】一年一度的MWC已经于当地时间2月26日在巴塞罗那正式开展,在本次MWC2024上全球各大厂商齐聚一堂,展出自家最新的技术与产品,其中中兴就参展本次MWC2024,并且展出了诸多面向企业端的产品,而旗下的
近日,联发科在MWC 2024(2024 世界移动通信大会)上展出了一系列令人瞩目的AI和移动通信技术突破,以“连接AI宇宙”(Connecting the AI-verse)的展厅吸引了无数业界精英和媒体的目光。特别是其现场的生成式AI技术
虽然目前国内已经有不少厂商入局折叠屏产品,但是努比亚却迟迟没有入局。不过在近日举办的MWC 2024展会上,努比亚发布了自家首款折叠屏手机——努比亚Flip,预计国内很快也会上市。
MWC 2024正在西班牙巴塞罗那举办,和往年一样,荣耀这次依旧携众多新产品、新技术参会。荣耀Magic6 Pro、荣耀Magic V2 RSR保时捷设计的机型在海外正式发布,并且还展示了魔法大模型、任意门等诸多新技术。
MWC 2024正在西班牙巴塞罗那如火如荼地举行,其中小米也参加了今年的大会,在会上发布了在国内大受欢迎的小尺寸旗舰——小米14。值得一提的是,高通公司CEO安蒙甚至亲临发布会现场为这款机型助阵。
《原神》是一直以来在机圈深受欢迎的游戏,在充满幻想的提瓦特大陆上,你可以邂逅不少性格迥异、能力独特的伙伴。而一加Ace系列一直就拥有非常强烈的电竞属性,也是畅玩《原神》的热门机型,而在本月,一加Ace 3将推
有不少网友发现,今年新机的发布时间相对于往年大幅提前,很多厂商在春节之前密集发布了自己最新的中高端机型,给人一种年后没什么新机可发了的感觉。不过魅族全新的大杯机型——魅族21 PRO非常值得期待,魅族科技也
2022年7月,小米12S Ultra正式发布,这款产品率先将1英寸大底主摄引入到移动影像领域,同时凭借鲜明的徕卡影调给人留下深刻的印象,同时这款产品也被视为了影像旗舰地位的机型。如果从那时算起,到现在差不多已经快过
随着智能手机的日益普及和智能化进程的加速,智能穿戴设备成为了人们关注的焦点。各大智能手机厂商纷纷进军智能穿戴市场,试图在这一新兴领域抢占先机。
早在去年秋天,HyperOS操作系统发布的时候,小米便勾勒出了“人车家全生态”的美好蓝图,而在这其中,小米的多终端统一战略是核心,目前已经有不少小米产品预装或者接受到了HyperOS操作系统的推送,在过去几个月的时
今年雷军将把更多的精力放在小米汽车上,所以接下来的手机业务将由刚刚兼任小米品牌总经理卢伟冰接管。同时雷军也在微博上表示小米2024年开年旗舰——小米14 Ultra即将在近期发布,并且将有卢伟冰进行讲解。另外,卢
新的一年有龙则灵,有愿必达。自1月19日起,荣耀加码“新年荣耀,一起成龙”年货节,在全国荣耀线下门店上线了“新年许愿处”、“龙运当头”等趣味活动,吸引大批消费者到店打卡许愿,戴龙头迎好运。与此同时,为了回
小米在官网微博中已经透露了关于小米14 Ultra信息,所以新机上市应该不会太晚。根据德国莱茵的官方消息,目前小米14 Ultra(型号为24030PN60G)获得了莱茵无频闪认证,表明这款手机可以有效减轻屏幕给用户带来的视觉疲
2月22日,上海广播电视台与华为举办鸿蒙合作签约仪式,宣布其官方客户端看看新闻APP将基于HarmonyOS NEXT鸿蒙星河版启动鸿蒙原生应用开发,为用户提供更加极致的新闻资讯服务体验。此次合作标志着上海广播电视台成为全国