迪米特法则(Law of Demeter, LoD)

别名:最少知识原则(Least Knowledge Principle)
核心思想:一个对象应尽可能少地与其他对象发生交互,只与直接的朋友(成员变量、方法参数、方法返回值中的对象)通信,避免依赖间接的类。


原理详解

  1. 直接朋友的定义

    • 当前对象的成员变量。
    • 当前对象方法的参数。
    • 当前对象方法的返回值。
    • 当前对象方法中创建的对象(不推荐,但允许)。
  2. 禁止链式调用
    避免出现 a.getB().getC().doSomething() 的调用形式,这种“火车残骸式”代码会增加耦合性。

  3. 目标

    • 降低耦合:减少模块间的依赖,提升代码可维护性。
    • 提高封装性:隐藏内部实现细节,仅暴露必要接口。

应用案例

场景1:订单系统获取用户配送地址
错误设计(违反迪米特法则)
public class User {private Order order;public Order getOrder() { return order; }
}public class Order {private Address shippingAddress;public Address getShippingAddress() { return shippingAddress; }
}public class Address {private String city;public String getCity() { return city; }
}// 客户端代码:链式调用(直接访问深层对象)
User user = new User();
String city = user.getOrder().getShippingAddress().getCity();

问题

  • 客户端需要了解 UserOrderAddress 的内部结构。
  • 修改 OrderAddress 的结构会影响客户端代码。
正确设计(遵循迪米特法则)
public class User {private Order order;public String getShippingCity() {return order.getShippingCity(); // 委托给 Order 类}
}public class Order {private Address shippingAddress;public String getShippingCity() {return shippingAddress.getCity(); // 委托给 Address 类}
}public class Address {private String city;public String getCity() { return city; }
}// 客户端代码:仅与直接朋友交互
User user = new User();
String city = user.getShippingCity();

优势

  • 客户端只需与 User 交互,无需了解 OrderAddress 的细节。
  • 修改 OrderAddress 的结构不会影响客户端代码。

场景2:文件系统目录结构遍历
错误设计(违反迪米特法则)
public class Directory {private List<File> files;private List<Directory> subDirectories;public List<File> getFiles() { return files; }public List<Directory> getSubDirectories() { return subDirectories; }
}// 客户端代码:直接遍历深层结构
public void printAllFiles(Directory root) {for (File file : root.getFiles()) {System.out.println(file.getName());}for (Directory subDir : root.getSubDirectories()) {printAllFiles(subDir); // 递归调用暴露内部结构}
}

问题

  • 客户端需要了解目录的递归结构,耦合度高。
正确设计(遵循迪米特法则)
public class Directory {private List<File> files;private List<Directory> subDirectories;// 封装遍历逻辑,客户端无需了解内部结构public void traverseFiles(Consumer<File> fileConsumer) {files.forEach(fileConsumer);subDirectories.forEach(subDir -> subDir.traverseFiles(fileConsumer));}
}// 客户端代码:仅调用高层方法
Directory root = new Directory();
root.traverseFiles(file -> System.out.println(file.getName()));

优势

  • 客户端仅依赖 DirectorytraverseFiles 方法,不关心内部实现。
  • 目录结构的遍历逻辑被封装,修改不影响客户端。

迪米特法则的实践意义

  1. 减少耦合:模块间通过接口通信,而非直接操作内部对象。
  2. 提升可维护性:修改一个类的内部结构时,无需调整其他模块。
  3. 增强可测试性:依赖越少,单元测试越容易隔离和模拟。

常见违反场景及修复

违反场景修复方法
链式调用(a.getB().getC()封装中间调用,提供高层接口(如 a.getC()
暴露集合内部结构返回不可变集合或迭代器,避免直接操作
方法参数传递复杂对象拆分参数为基本类型或接口

总结

迪米特法则通过限制对象间的交互范围,推动代码向高内聚、低耦合的方向演进。其核心是封装委托,适用于任何需要降低依赖关系的场景,尤其在大型系统或模块化架构中价值显著。

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

如若内容造成侵权/违法违规/事实不符,请联系英文站点网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

python获取AB直线间任意一点经纬度

获取AB直线间任意一点经纬度 1、目标 已知A点经纬度,距离;B点经纬度,距离,如果C点在AB之间,且知道C点距离,求C点的经纬度信息。 目标:在AB这条直线上,根据给定的距离(从A点开始沿直线到某点的距离)来求该点的经纬度。 2、方法 首先计算AB的总长度(大圆距离),…

Android实战——系统字体库加载流程

Android 系统字体库指的是在Android设备上用于显示文本的字体集合。随着Android系统的更新,其对字体的支持也日益增强,允许开发者和用户更灵活地定制界面文字显示。 一、字体库介绍 1、字体库文件 字体库文件是指存储字体数据的文件,这些文件包含了创建文本字符所需的所有…

嵌入式乐鑫音频项目“无声”问题深度调试复盘与方法论总结

前言&#xff1a;一场典型的“工程师寻踪之旅” 本次调试始于一个看似简单却极其顽固的问题&#xff1a;在一个基于乐鑫ESP-ADF&#xff08;音频开发框架&#xff09;的DuerOS示例项目中&#xff0c;移植到M5Stack ATOMIC Echo Base硬件上后&#xff0c;程序能够成功编译、烧录…

地下安全防线:电缆通道防外破地钉如何守护城市隐形生命线

在繁华都市的柏油马路之下、在静谧乡村的泥土深处&#xff0c;纵横交错的地下管线如同城市与乡村的 “隐形生命线”&#xff0c;承载着电力输送、供水供气、通信传输等重要功能&#xff0c;默默维系着现代社会的正常运转。然而&#xff0c;这条 “生命线” 正面临着诸多潜在威胁…

linux时间同步方案

yum install chrony -y # 配置 chrony 使用国内服务器 sed -i s/^pool.*pool.ntp.org/#&/ /etc/chrony.conf cat >> /etc/chrony.conf <<EOF server ntp.aliyun.com iburst server ntp.tencent.com iburst server ntp.ntsc.ac.cn iburst server time1.cloud.t…

C语言笔记(鹏哥)上课板书+课件汇总(KMP算法的动态规划简易处理+字符函数和字符串函数)

一、目录 kmp动态规划简易处理next数组字符函数与字符串函数 一、目录二、引言C语⾔标准库中提供了⼀系列库函数 三、字符分类函数&#xff08;字符相关的函数&#xff09;推荐一个网站 四、字符转换函数&#xff08;字符相关的函数&#xff09;五、strlen&#xff08;字符串相…

Java大模型开发入门 (13/15):拥抱官方标准 - Spring AI框架入门与实践

前言 到目前为止&#xff0c;我们整个系列的旅程都是在功能强大的LangChain4j框架上构建的。它就像一个装备齐全的“瑞士军刀”&#xff0c;为我们提供了构建RAG和Agents所需的所有底层和高层工具。 然而&#xff0c;在Java企业级开发的世界里&#xff0c;有一个名字我们永远…

Github搜索案例

今天的内容是这个案例的实现&#xff0c;以及其中涉及到的内容&#xff0c;需要全部掌握&#xff0c;比如ref&#xff0c;受控组件&#xff0c;props在组件之中的传递&#xff0c;以及Pubsub包的使用这些前端React框架有关的内容。现在进入正题 1.github搜索案例&#xff08;a…

Vue3学习(生命周期,hooks,axios的简单讲解)

一&#xff0c;前言 继续努力&#xff0c;南方见。 二&#xff0c;生命周期 1.对生命周期的理解 例如&#xff1a;人的生命周期&#xff0c;出生&#xff0c;经历&#xff0c;死亡 组件的话就是&#xff0c;创建&#xff0c;挂载&#xff0c;更新&#xff0c;销毁。***在特…

Pytorch实战四 基于 VGG net 搭建一个串联的神经网络结构

系列文章目录 文章目录 系列文章目录前言一、VGG类的搭建1.源码2.初始化类2.1 初始化函数2.2 前向传播函数 forward(self,x) 二、卷积补充卷积 前言 对于标准的 VGG net 输入图像的尺寸是 24 x 24,进行 32 维的下采样之后得到一个 7 x 7 的特征图&#xff0c;然后用 FC 层完成分…

大学专业解读——计算机

我们继续&#xff0c;讲讲排名第二流行的新工科专业——计算机。说到计算机&#xff0c;可能所有人都知道&#xff0c;但具体到细分的专业类别&#xff0c;除了计算机科学&#xff0c;其实大多数人都是不了解的。 序&#xff1a; 计算机主要有如下几个专业&#xff1a; 计算机…

Bootstrap 5学习教程,从入门到精通, Bootstrap 5 列表组(List Group)语法知识点及案例(14)

Bootstrap 5 列表组(List Group)语法知识点及案例 一、列表组基础语法 列表组是Bootstrap中用于显示一系列内容的灵活组件&#xff0c;常用于显示菜单、导航或任何项目列表。 基本列表组结构 <ul class"list-group"><li class"list-group-item&quo…

FPGA基础 -- Verilog 命名事件

Verilog 的“命名事件&#xff08;Named Events&#xff09;”机制 进行一次系统、专业的培训。该机制在 Verilog 中是比较冷门但重要的仿真控制特性&#xff0c;主要用于 模块间同步、行为仿真触发、事件通信&#xff0c;在复杂的 Testbench、行为模型中尤为重要。 一、命名事…

《Go语言圣经》结构体

《Go语言圣经》结构体 一、结构体指针的高效应用 在处理大型结构体时&#xff0c;为避免内存复制&#xff0c;通常使用指针传递和返回结构体&#xff1a; // 通过指针传入结构体&#xff0c;避免值拷贝 func Bonus(e *Employee, percent int) int {return e.Salary * percen…

Ascend上如何进行带宽测试

1 工具安装 1.1 下载链接 https://www.hiascend.com/developer/download/community/result?moduledl%2Bcann 1.2 安装指令&#xff1a; ./Ascend-mindx-toolbox_{version}_linux-{arch}.run --install设置环境变量&#xff1a; source /usr/local/Ascend/toolbox/set_env.…

生产BUG集

磁盘达到阈值导致ES无法删除数据 method [POST], host [http://xx.xxx.xxx.xxx:9200], URI [/security_event/_delete_by_query?slices1&requests_per_second-1&ignore_unavailablefalse&expand_wildcardsopen&allow_no_indicestrue&ignore_throttledtru…

基于FastAPI与Selenium的智能开关状态管理系统实践

引言 在工业物联网&#xff08;IIoT&#xff09;与自动化控制场景中&#xff0c;设备状态的实时监控与自然语言指令执行是提升效率的关键。本文将介绍一种基于 FastAPI 和 Selenium 的智能设备状态管理系统&#xff0c;通过大语言模型&#xff08;LLM&#xff09;解析用户指令…

主体和债项均为“AAA”等级 海尔消金发行10亿金融债

6月18日&#xff0c;继年内发行ABS、落地ESG挂钩银团贷后&#xff0c;海尔消费金融&#xff08;以下简称“海尔消金”&#xff09;在金融市场上又迈出重要一步&#xff0c;成功簿记发行2025年首期规模达10亿元金融债&#xff0c;且主体信用等级仍为“AAA”。这一举措为海尔消金…

n8n:轻松自动化您的工作流

借助开源自动化利器 n8n&#xff0c;释放重复劳动的生产力&#xff01; 引言 n8n 是一款免费、开源的工作流自动化工具&#xff0c;致力于帮助开发者和团队通过连接各种应用和服务&#xff0c;实现重复任务的自动化处理。 它由 Jan Oberhauser 于 2019 年在德国柏林创建&…

Angular--Hello(TODO)

最近有个小错误&#xff0c;因为最近还是在看thingsboard&#xff0c;最近终于看到前端的代码&#xff0c;突然发现怎么全是ts的文件&#xff0c;仔细一看原来并不是之前认为的AngularJS&#xff0c;而是Angular。。。我tm真的无语了&#xff0c;又要去重新学。。。 Angular的…