设计模式学习笔记 - 设计原则 - 8.迪米特法则(LOD)

news2024/9/27 19:25:09

前言

迪米特法则,是一个非常实用的原则。利用这个原则,可以帮我们实现代码的 “高内聚、松耦合”。

围绕下面几个问题,来学习迪米特原则。

  • 什么是 “高内聚、松耦合”?
  • 如何利用迪米特法则来实现 高内聚、松耦合?
  • 哪些代码设计是明显违背迪米特法则的?该如何重构?

什么是 “高内聚、松耦合”?

“高内聚、松耦合”是一个非常重要的思想,能有效地提高代码的可读性和可维护性,缩小功能改动导致的代码范围改动。

很多设计原则都已实现代码的“高内聚、松耦合”为目的,比如单一职责原则、基于接口而非实现编程。

在这个设计思想中,“高内聚” 用来指导类本身的设计,“松耦合” 用来指导类与类之间依赖关系的设计。不过,这两种并非独立不相干。高内聚有助于松耦合,松耦合有需要高内聚的支持。

什么是“高内聚”呢?

所谓高内聚,就是指相近的功能应该放到同一个类中,不相近的功能不要放到同一个类中

相近的功能往往会被同时修改,放到同一个类中,修改会比较集中,代码容易维护。实际上,我们前面讲过的单一职责原则是实现代码高内聚的非常有效的设计原则。

什么是“松耦合”?

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

前面讲的依赖注入接口隔离基于接口而非实现编程,以及今天的迪米特法则,都是为了实现代码的松耦合。

“内聚” 和 “耦合” 之间的关系

“高内聚” 有助于 “松耦合”,同理 “低内聚” 也会导致 “紧耦合”。

在这里插入图片描述
上图的左边的代码结构是 “高内聚、松耦合”,右边部分是 “低内聚、紧耦合”。

  • 左边部分的代码设计中,类的粒度较小,每个类的职责比较单一。相近的功能都放到了一个类中,不相近的功能被分割到了多个类中。这样类更加独立,代码的内聚性更高。因为职责单一,所以每个类被依赖的类就会比较少,代码低耦合。一个类的修改,只会影响到一个依赖类的代码改动。我们只需要测试依赖类是否还能正常工作就行了。
  • 右边部分的代码设计中,类粒度较大,低内聚,功能大而全,不相近的功能放到一个类中。这就导致很多其他类都依赖这个类。当修改这个类的某一个功能代码时,会影响依赖它的多个类。我们需要测试这三个依赖类是否正常工作。这也就是所谓的 “牵一发而动全身”。

另外,从图中我们可以看出,高内聚、低耦合的代码结构更加简单、清晰,相应地,可维护性和可读性要好很多。

“迪米特法则”理论描述

迪米特法则,Law Of Demeter,缩写是 LOD,它也叫做最小知识原则。

这个原则的英文定义:

Each unit should haval only limited knwoledge about other units: only units “closely” related to the current unit. Or: Ecah unit should only talk to its friends; Don’t talk to strangers.

翻译:每个模块只应该了解与它关系紧密的模块的有限知识。或者说,每个模块只和自己的朋友 “说话”,不和陌生人说话。

我们把其中的模块,替换成类,按照自己的意思理解下:不该有直接依赖关系的类,不要有依赖;有依赖关系的类,尽量只依赖必要的接口

从上面的描述中,我们可以看出,迪米特法则包含前后两部分,这两部分讲的是两件事,我们用两个实战案例讲解下。

理论解读与代码实战一 —— 不要有直接依赖关系的类,不要有依赖

我们举一个搜索引擎爬取网页的功能。代码中包含三个主要类。其中 NetworkTransporter 类负责底层网络通信,根据请求获取数据;HtmlDownloader 类用来通过 URL 获取网页;Document 表示网页文档,后续的网页内容抽取、分词、索引都是依次为处理对象。具体代码如下:

public class NetworkTransporter {
    // 省略属性和其他方法...
    public  Byte[] send(HtmlRequest htmlRequest) { /*...*/ }
}

public class HtmlDownloader {
    private NetworkTransporter transporter; // 通过构造函数注入...
    
    public Html downloadHtml(String url) {
        Byte[] rawHtml = transporter.send(new HtmlRequest(url));
        return new Html(rawHtml);
    }
}

public class Document {
    private Html html;
    private String url;

    public Document(String url) {
        this.url = url;
        HtmlDownloader downloader = new HtmlDownloader();
        this.html = downloader.downloadHtml(url);
    }
}

这段代码虽然能用,但是有较多的设计缺陷。

首先来看 NetworkTransporter。 作为一个底层网络通信类,我们希望它的功能尽可能统一,而不是只用于下载 HTML。所以,不应该直接依赖太具体的发送对象 HtmlRequest

该如何重构,才能让 NetworkTransporter 类符合迪米特法则呢?应该把 HtmlRequest 类中的 addresscontent 交给 NetworkTransporter ,而非直接把 HtmlRequest 交给 NetworkTransporter。按照这个思路,NetworkTransporter 重构之后的代码如下所示:

public class NetworkTransporter {
    // 省略属性和其他方法...
    public  Byte[] send(String address, Byte[] data) { /*...*/ }
}

在看下 HtmlDownloader。这个类的设计没有问题。不过因为修改了 NetworkTransporter,所以要对它做响应的改动。

public class HtmlDownloader {
    private NetworkTransporter transporter; // 通过构造函数注入...

    public Html downloadHtml(String url) {
        HtmlRequest htmlRequest = new HtmlRequest(url);
        Byte[] rawHtml = transporter.send(htmlRequest.getAddress(), htmlRequest.getContent().getBytes());
        return new Html(rawHtml);
    }
}

最后,看下 Document。 这个类的问题比较多,主要有三点。

  • 第一:构造函数中的 downloader.downloadHtml(url) 逻辑复杂,耗时长,不应该放到构造函数中,会影响代码的可测试性。
  • 第二:HtmlDownloader 在构造函数中,通过 new 来创建,违反了基于接口而非实现编程的设计思想,也会影响到代码的可测试性。
  • 第三:从业务角度上讲,Document 网页文档没必要依赖 HtmlDownloader,违背迪米特法则。

不过,Document 修改起来还是比较简单的,只要一处改动,就可以解决所有问题。

public class Document {
    private Html html;
    private String url;

    public Document(String url, Html html) {
        this.html = html;
        this.url = url;
    }
    //...
}

// 通过一个工厂创建 Document
public class DocumentFactory {
    private HtmlDownloader htmlDownloader;

    public DocumentFactory(HtmlDownloader htmlDownloader) {
        this.htmlDownloader = htmlDownloader;
    }

    public Document createDocument(String url) {
        Html html = htmlDownloader.downloadHtml(url);
        return new Document(url, html);
    }
}

理论解读与代码实战二 —— 有依赖关系的类,尽量只依赖必要的接口

现在在看下迪米特法则的后半部分:“有依赖关系的类,尽量只依赖必要的接口”。我们还是结合一个例子来讲解。下面这段代码非常简单, Serialization 类负责对象的序列化和反序列化。这个例子在之前单一职责原则笔记中,你可以结合一起看下。

public class Serialization {
    public String serialize(Object object) {
        String serializedResult = ...;
        // ...
        return serializedResult;
    }

    public Object deserialize(String str) {
        String deserializedResult = ...;
        // ...
        return deserializedResult;
    }
}

但看这个类没有一点问题。不过,若把它放到一定的应用场景里,那就还有继续优化的空间。 假设,在我们的项目中,有些类只用到了序列化操作,有些类去只用了反序列化操作。基于迪米特法则的后半部分 “有依赖关系的类,尽量只依赖必要的接口”,只用到序列化的那部分类不应该依赖反序列化接口。同理,只用到反序列化的那部分类不应该依赖序列化接口。

根据这个思路,将 Serialization 拆分成两个更小粒度的类,一个只负责序列化 Serializer,另一个只负责反序列化 Deserializer

public class Serializer {
    public String serialize(Object object) {
        String serializedResult = ...;
        // ...
        return serializedResult;
    }
}

public class Deserializer {
    public Object deserialize(String str) {
        String deserializedResult = ...;
        // ...
        return deserializedResult;
    }
}

不过,虽然拆分之后的代码更能满足迪米特法则,但却违背了高内聚的设计思想。高内聚要求相近的功能要放到同一个类中,这样可以方便功能修改的时候,修改的地方不至过于分散。对于刚刚的例子,如果我们修改了序列化的实现方式,比如从 JSON 换成了 XML,那反序列化的实现逻辑也需要修改。在未拆分的情况下,只要修改一个类即可。在拆分之后,需要修改两个类。显然,这种设计思路的代码改动范围变大了。

实际上,通过引入两个接口,就可以既不违背高内聚的设计思想,也不违背迪米特法则。具体的代码如下所示,实际上在讲解 “接口隔离原则”课程的时候,第三个例子就用了类似的思路。你可以结合着一块来看。

public interface Serializable {
    String serialize(Object object);
}

public interface Deserializable {
    Object deserialize(String str);
}

public class Serialization implements Serializable, Deserializable {
    @Override
    public String serialize(Object object) {
        String serializedResult = ...;
        // ...
        return serializedResult;
    }
    
    @Override
    public Object deserialize(String str) {
        String deserializedResult = ...;
        // ...
        return deserializedResult;
    }
}

public class DemoClass_1 {
    private Serializable serializer;
    
    public DemoClass_1(Serializable serializer) {
        this.serializer = serializer;
    }
    // ...
}

public class DemoClass_2 {
    private Deserializable deserializer;
    
    public DemoClass_2(Deserializable deserializer) {
        this.deserializer = deserializer;
    }
    // ...
}

尽管,还是要往 DemoClass_1 类的构造函数中,传入包含序列化和反序列化的 Serialization 实现类,但是我们依赖的 Serializable 接口只包含序列化操作, DemoClass_1 无法使用 Serialization 的发序列化接口,对反序列化操作无感知,这也就符合了迪米特法则后半部分所说的“依赖有限接口”的要求。

实际上,上面的代码思路,也体现了 “基于接口而非实现编程” 的设计原则,结合迪米特法则,我们可以总结出一个新的设计原则,那就是 “基于最小接口而非最大实现编程”。

辩证思考与灵活应用

Serialization 类只包含序列化和反序列化两个操作,只用到序列化操作的使用者,即便能够感知到反序列化接口,问题也不大。为了满足迪米特法则,我们将一个非常简单的类,拆分出两个接口,是否有过度设计的意思呢?

设计原则本身没错,只有是否用对之说。不要为了应用设计原则而应用设计原则,我们在应用设计原则时,一定要具体问题具体分析。

对于刚刚的 Serialization 来说,只包含两个操作,确实没有太大必要拆分成两个接口。但是,如果我们对 Serialization 类添加更多的功能,实现更多更好的序列化和反序列化函数,我们来重新思考下这个问题。修改后的代码如下所示:

public class Serialization {
    public String serialize(Object object) { /*...*/ }
    public String serialize(Map map) { /*...*/ }
    public String serialize(List list) { /*...*/ }
    
    public Object deserialize(String str) { /*...*/ }
    public Map deserialize(String str) { /*...*/ }
    public List deserialize(String str) { /*...*/ }
}

在这种场景下,第二种设计思路更好些。因为基于之前的应用常见来说,大部分代码只需要用到序列化的功能,这部分使用者,没有必要了解反序列化知识,而修改之后的 Serialization 的反序列化知识从一个变成三个。 一旦,反序列化操作有代码改动,我们都需要检查、测试所有依赖 Serialization 类的代码是否还能正常工作。为了减少耦合和测试工作量,我们应该按照迪米特法则,将反序列化和序列化功能隔离开。

总结

1.如何理解“高内聚、松耦合”

“高内聚、松耦合” 是一个非常重要的设计思想,能有效提高代码的可读性、可维护性,缩小功能改动导致的代码改动范围。“高内聚” 用来指导类本身的设计,“低耦合” 用来指导类与类之间的依赖关系的设计。

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

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

2.如何理解“迪米特法则”

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

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

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

【python debug】python常见编译问题解决方法_2

序言 记录python使用过程中碰到的一些问题及其解决方法上一篇:python常见编译问题解决方法_1 1. PermissionError: [Errno 13] Permission denied: ‘/lostfound’ 修改前: 修改后(解决): 此外,可能文件夹…

开发者38万+,鸿蒙开发岗为何却无人敢应聘?

鸿蒙校园公开课已走进135家高校,305所高校学生参与鸿蒙活动,286家企业参加鸿蒙生态学堂,38万开发者通过鸿蒙认证。 居上华为官方是说有通过鸿蒙开发者认证的已有38万。具体有多少开发者并没有明确表示。除此之外还有200家头部应用加速鸿蒙原…

机器人 标准DH与改进DH

文章目录 1 建立机器人坐标系1.1 连杆编号1.2 关节编号1.3 坐标系方向2 标准DH(STD)2.1 确定X轴方向2.2 建模步骤2.3 变换顺序2.4 变换矩阵3 改进DH(MDH)3.1 确定X轴方向3.2 建模步骤3.3 变换顺序3.4 变换矩阵4 标准DH与改进DH区别5 Matlab示例参考链接1 建立机器人坐标系 1.1…

Java二叉树(1)

🐵本篇文章将对二叉树的相关概念、性质和遍历等知识进行讲解 一、什么是树 在讲二叉树之前,先了解一下什么是树:树是一种非线性结构,其由许多节点和子节点组成,整体形状如一颗倒挂的树,比如下图&#xff1…

探索设计模式的魅力:备忘录模式揭秘-实现时光回溯、一键还原、后悔药、历史的守护者和穿越时空隧道

​🌈 个人主页:danci_ 🔥 系列专栏:《设计模式》 💪🏻 制定明确可量化的目标,并且坚持默默的做事。 备忘录模式揭秘-实现时光回溯、一键还原、后悔药和穿越时空隧道 文章目录 一、案例场景&…

Docker架构概述

Docker是基于Go语言实现的开源容器项目,能够把开发的应用程序自动部署到容器的开源的应用容器引擎。Docker的构想是要实现"Build, Ship and Run Any App, Anywhere",即通过对应用的封装(Packaging)、分发(Distribution)、部署(Deployment)、运…

Autosar Appl介绍

AUTOSAR架构中的应用层 AUTOSAR 应用层构成AUTOSAR 架构中的最顶层,被认为对所有车辆应用至关重要。AUTOSAR 标准使用“组件”概念指定应用层实现。 在谈论应用层实现时,应该考虑的三个最重要的部分是: AUTOSAR 应用软件组件这些组件的 AUTOSAR 端口AUTOSAR 端口接口 AUTOS…

LeetCode受限条件下可到达节点的数目

题目描述 现有一棵由 n 个节点组成的无向树,节点编号从 0 到 n - 1 ,共有 n - 1 条边。 给你一个二维整数数组 edges ,长度为 n - 1 ,其中 edges[i] [ai, bi] 表示树中节点 ai 和 bi 之间存在一条边。另给你一个整数数组 restr…

Game With Sticks

最近思维实在是不活跃。。。。。。 题目链接&#xff1a;Submit - Codeforces 解题思路&#xff1a; 如果n > m,交换 直接判断n就行&#xff0c;偶数M赢&#xff0c;奇数A赢 下面是c代码&#xff1a; #include<iostream> using namespace std; int main() {int n…

iZotope RX 10:专业音频修复,尽在指尖 mac/win版

iZotope RX 10是一款革命性的音频修复和增强软件&#xff0c;它为音频专业人士、电影制片人、音乐制作人和广播工作者提供了无与伦比的工具集&#xff0c;以处理和改善各种音频问题。 iZotope RX 10 软件获取 RX 10的核心是其先进的音频分析和修复算法&#xff0c;这些算法能够…

经典目标检测网络Yolo——原理部分

目标检测问题 分为两个子问题: 找到图片中哪些位置、哪些区域含有目标对象识别这些区域中的目标对象是什么基于CNN的目标检测算法能够很好的解决第二个问题,在一张图片仅含一个对象,且该对象占据了整张图片绝大部分面积时,基于CNN的对象识别算法具有很高的准确率。 一种定…

基于YOLOv的目标追踪与无人机前端查看系统开发

一、背景与简介 随着无人机技术的快速发展&#xff0c;目标追踪成为无人机应用中的重要功能之一。YOLOv作为一种高效的目标检测算法&#xff0c;同样适用于目标追踪任务。通过集成YOLOv模型&#xff0c;我们可以构建一个无人机前端查看系统&#xff0c;实现实时目标追踪和可视化…

搜素题目(蓝桥杯 C++ 代码+注解)

目录 题目一&#xff08;小朋友崇拜圈&#xff09;&#xff1a; 代码&#xff1a; 题目二&#xff08;穿越雷区&#xff09;&#xff1a; 代码&#xff1a; 题目三&#xff08;分考场&#xff09;&#xff1a; 代码&#xff1a; 题目四&#xff08;受伤的皇后&#xff09…

c++之旅——第三弹

大家好啊&#xff0c;这里是c之旅第三弹&#xff0c;跟随我的步伐来开始这一篇的学习吧&#xff01; 如果有知识性错误&#xff0c;欢迎各位指正&#xff01;&#xff01;一起加油&#xff01;&#xff01; 创作不易&#xff0c;希望大家多多支持哦&#xff01; 一.命名空间;…

基于springboot实现图书馆管理系统项目【项目源码+论文说明】计算机毕业设计

基于springboot实现图书馆管理系统演示 摘要 电脑的出现是一个时代的进步&#xff0c;不仅仅帮助人们解决了一些数学上的难题&#xff0c;如今电脑的出现&#xff0c;更加方便了人们在工作和生活中对于一些事物的处理。应用的越来越广泛&#xff0c;通过互联网我们可以更方便地…

OpenCV 4基础篇| OpenCV图像的拼接

目录 1. Numpy (np.hstack&#xff0c;np.vstack)1.1 注意事项1.2 代码示例 2. matplotlib2.1 注意事项2.2 代码示例 3. 扩展示例&#xff1a;多张小图合并成一张大图4. 总结 1. Numpy (np.hstack&#xff0c;np.vstack) 语法结构&#xff1a; retval np.hstack(tup) # 水平…

从嵌入式Linux到嵌入式Android

最近开始投入Android的怀抱。说来惭愧&#xff0c;08年就听说这东西&#xff0c;当时也有同事投入去看&#xff0c;因为恶心Java&#xff0c;始终对这玩意无感&#xff0c;没想到现在不会这个嵌入式都快要没法搞了。为了不中年失业&#xff0c;所以只能回过头又来学。 首先还是…

考研数学——高数:微分方程

一、一阶线性微分方程 两种形式&#xff1a; 非齐次&#xff1a; 齐次&#xff1a; 推导过程 推导公式的过程一般由特殊到一般&#xff1a;所以先求解齐次方程的解 &#xff08;然后对等式两边同时积分&#xff09; 再来求非齐次方程的解&#xff0c;由…

基于springboot实现保险信息网站系统项目【项目源码+论文说明】计算机毕业设计

基于springboot实现保险信息网站系统演示 摘要 随着互联网的不断发展&#xff0c;现在人们获取最新资讯的主要途径来源于网上新闻&#xff0c;当下的网上信息宣传门户网站的发展十分的迅速。而保险产品&#xff0c;作为当下人们非常关注的一款能够给人们带来医疗、生活、养老或…

VUE3:省市区联级选择器

一、实现效果 二、代码展示 <template><div class"page"><select v-model"property.province"><option v-for"item in provinces" :key"item">{{ item }}</option></select><select v-model&…