【设计原则】迪米特法则(LoD):降低耦合的设计智慧

深入理解迪米特法则

    • 一、什么是迪米特法则?
    • 二、违反迪米特法则的示例
    • 三、遵循迪米特法则的重构
      • 重构方案一:封装间接访问
      • 重构方案二:通过接口解耦
    • 四、关键改进点分析
    • 五、迪米特法则的应用场景
    • 六、最佳实践建议
    • 七、总结
    • 八、扩展思考

一、什么是迪米特法则?

迪米特法则(Law of Demeter,LoD),又称最少知识原则(Least Knowledge Principle),是面向对象设计中的重要原则之一。其核心思想是:

一个对象应该对其他对象保持最少的了解
只与直接的朋友通信
不要和"陌生人"说话

该原则通过降低类之间的耦合度来提高系统的可维护性和代码质量。


二、违反迪米特法则的示例

// 城市类,表示城市信息
public class City
{public string Name { get; set; }  // 城市名称属性
}// 地址类,表示地址信息,包含城市信息
public class Address
{public City City { get; set; }  // 城市对象属性
}// 客户类,表示客户信息,包含地址信息
public class Customer
{public Address Address { get; set; }  // 地址对象属性
}// 订单类,表示订单信息,需要获取客户的城市信息
public class Order
{private Customer _customer;  // 客户对象// 构造函数,初始化订单时需要传入客户对象public Order(Customer customer){_customer = customer;}// 打印配送城市的方法public void PrintShippingCity(){// 违反迪米特法则:// Order类直接访问了Customer的内部结构(Address),// 并通过Address访问了City的内部结构(Name),// 形成了Customer->Address->City的链式调用。// 这种设计导致Order类与Customer、Address、City类紧密耦合。Console.WriteLine($"Shipping city: {_customer.Address.City.Name}");}
}

问题分析:

  1. Order类直接访问了Customer的内部结构:Order类需要了解Customer的内部实现细节(Address属性)。
  2. 需要了解Customer->Address->City的完整链式关系:Order类依赖于多层嵌套的对象结构。
  3. 任何一个中间环节的变化都会影响Order类:如果Address或City的结构发生变化,Order类也需要修改。
  4. 耦合度过高,维护成本大:代码的可维护性和扩展性较差。

三、遵循迪米特法则的重构

重构方案一:封装间接访问

// 改进后的客户类,封装了获取城市名称的逻辑
public class Customer
{private Address _address;  // 将Address属性改为私有字段,增强封装性// 封装获取城市名称的方法// 目的:隐藏内部数据结构细节,避免外部类直接访问Address和Citypublic string GetCityName(){// 将链式调用封装在Customer内部// 外部类只需调用GetCityName方法,无需知道Address和City的存在return _address.City.Name;}
}// 改进后的订单类
public class Order
{private Customer _customer;  // 客户对象// 构造函数,初始化订单时需要传入客户对象public Order(Customer customer){_customer = customer;}// 打印配送城市的方法public void PrintShippingCity(){// 现在Order类只需与Customer直接交互,// 不再需要知道Address和City的存在。// 通过调用GetCityName方法获取城市名称,符合迪米特法则。Console.WriteLine($"Shipping city: {_customer.GetCityName()}");}
}

重构方案二:通过接口解耦

// 城市信息提供接口,定义获取城市名称的标准方法
public interface ICityInfoProvider
{string GetCityName();  // 抽象方法,由具体类实现
}// 客户类实现ICityInfoProvider接口
public class Customer : ICityInfoProvider
{private Address _address;  // 地址对象// 实现ICityInfoProvider接口的方法public string GetCityName(){// 封装获取城市名称的逻辑return _address.City.Name;}
}// 改进后的订单类,依赖ICityInfoProvider接口
public class Order
{private ICityInfoProvider _cityInfo;  // 依赖抽象接口,而非具体实现// 构造函数,通过依赖注入获取城市信息提供者public Order(ICityInfoProvider cityInfo){_cityInfo = cityInfo;}// 打印配送城市的方法public void PrintShippingCity(){// 通过接口进行交互,完全解耦具体实现类。// 可支持任何实现了ICityInfoProvider的对象,提高了代码的灵活性和可扩展性。Console.WriteLine($"Shipping city: {_cityInfo.GetCityName()}");}
}

四、关键改进点分析

  1. 降低耦合度

    • Order类不再依赖具体的地址结构(Address和City)。
    • 只需关注直接关联的Customer类或ICityInfoProvider接口。
  2. 增强封装性

    • 隐藏了Customer的内部实现细节(Address和City)。
    • 将地址信息的获取逻辑封装在Customer内部。
  3. 提高可维护性

    • 当地址结构变化时,只需修改Customer类,Order类不需要任何修改。
    • 减少了代码的修改范围和影响。
  4. 接口抽象

    • 使用接口进一步解耦依赖关系,提高了代码的可测试性和扩展性。
    • 支持依赖注入,便于单元测试和模块替换。

五、迪米特法则的应用场景

  1. 分层架构设计:各层之间通过接口通信,避免直接依赖具体实现。
  2. 模块间通信:模块之间通过定义清晰的接口交互,减少耦合。
  3. 服务接口设计:微服务架构中,服务之间通过API通信,避免直接访问内部数据结构。
  4. 组件化开发:组件之间通过抽象接口交互,提高组件的复用性。
  5. 微服务架构中的服务边界划分:明确服务职责,避免服务之间的过度依赖。

六、最佳实践建议

  1. 遵循单一职责原则:每个类只关注自己的核心职责,避免承担过多功能。
  2. 谨慎使用链式调用:避免a.b.c.d形式的访问,减少类之间的直接依赖。
  3. 优先使用组合而非继承:通过组合方式实现代码复用,降低父子类之间的耦合。
  4. 合理使用中间层:通过适配器或门面模式封装复杂调用,简化外部类的使用。
  5. 接口隔离原则:定义细粒度的专用接口,避免接口臃肿。

七、总结

迪米特法则通过限制对象之间的交互范围,带来了以下优势:

优势说明
降低耦合度减少类之间的直接依赖
提高封装性隐藏实现细节
增强可维护性修改影响范围可控
提升代码质量更清晰的类职责划分
方便单元测试依赖关系简单明确

注意事项:避免过度设计,在简单场景中可以直接访问属性。应根据实际项目需求和复杂度权衡设计粒度。


八、扩展思考

当处理复杂系统时,可以结合其他设计模式:

  • 门面模式(Facade):为子系统提供统一接口,简化外部调用。
  • 中介者模式(Mediator):通过中介对象协调多个对象之间的交互,减少直接依赖。
  • 适配器模式(Adapter):转换接口格式,使不兼容的类能够协同工作。

正确应用迪米特法则能使系统更灵活,更易应对需求变化,是构建高质量软件系统的重要保障。

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

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

相关文章

嵌入式八股RTOS与Linux---前言篇

前言 Linux与RTOS是校招八股的时候很喜欢考察的知识,在这里并没有把两个操作系统完全的独立开去讲,放在一起对比或许可能加深印象。我们讲Linux的内核有五部分组成:进程调度、内存管理、文件系统、网络接口、进程间通信,所以我也将从这五方面出发 中断管理去对比和RTOS的不同。…

centos 8安装及相关操作

安装centos 8 在VMware workstation中安装 UEFI对比BIOS有更快的启动速度、支持更大容量硬盘及 GPT 分区、图形化操作界面更友好、安全性更高、对新操作系统支持更好、硬件兼容性不断增强以及扩展性更好等。 按回车确定 重置root管理员密码 这样进入到紧急救援模式 mount -o r…

2025最新版Windows通过GoLand远程连接Linux构建Go项目保姆级教学

以Ubuntu24.04和GoLand2024.1.6为例子,演示如何在Windows上通过GoLand远程连接Linux进行Go编程。 通过go version指令可以发现当前Ubuntu系统没有安装go。 go version 通过指令安装go,其他系统可以通过wget安装,要指定安装的具体go版本&…

多元时间序列预测的范式革命:从数据异质性到基准重构

本推文介绍了一篇来自中国科学院计算技术研究所等机构的论文《Exploring Progress in Multivariate Time Series Forecasting: Comprehensive Benchmarking and Heterogeneity Analysis》,发表在《IEEE Transactions on Intelligent Transportation Systems》。论文…

开源PACS(dcm4che-arc-light)部署教程,源码方式

目录 文件清单下载地址安装概述OpenLDAP、Apache Directory StudioWildflydcm4che 安装部署MySQL源码编译dcm4cheedcm4chee-arc-light OpenLDAP安装ApacheDirectoryStudio安装配置WildFly服务器 部署完成 文件清单 下载地址 Apache directory studio - linkOpenLDAP - linkdcm…

PySide(PyQt),使用types.MethodType动态定义事件

以PySide(PyQt)的图片项为例,比如一个视窗的场景底图是一个QGraphicsPixmapItem,需要修改它的鼠标滚轮事件,以实现鼠标滚轮缩放显示的功能。为了达到这个目的,可以重新定义一个QGraphicsPixmapItem类,并重写它的wheelE…

深度学习 Deep Learning 第1章 深度学习简介

第1章 深度学习简介 概述 本章介绍人工智能(AI)和深度学习领域,讨论其历史发展、关键概念和应用。解释深度学习如何从早期的AI和机器学习方法演变而来,以及如何有效解决之前方法无法应对的挑战。 关键概念 1. 人工智能的演变 …

简述下npm,cnpm,yarn和pnpm的区别,以及跟在后面的-g,--save, --save-dev代表着什么

文章目录 前言一、npm,cnpm,yarn和pnpm的基本介绍和特点1.npm (Node Package Manager)2. Yarn3. cnpm (China npm)4. pnpm 二、简述npm和pnpm 的存储方式和依赖数1.存储方式2.依赖树 三、两者依赖树的差异导致结果的对比四、简单说说-g,--sav…

vue3系列:vite+vue3怎么配置通过ip和端口打开浏览器

目录 1.前言 2.修改前的 3.修改后的 4.效果 5.其他 1.前言 想要使用IP端口号的方式访问页面,结果无法访问 查了些资料,原来是vite.config.js需要加一些配置才能让他通过IP访问,默认的只能localhost:端口号访问 2.修改前的 使用vue3默认…

使用yolov8+flask实现精美登录界面+图片视频摄像头检测系统

这个是使用flask实现好看登录界面和友好的检测界面实现yolov8推理和展示,代码仅仅有2个html文件和一个python文件,真正做到了用最简洁的代码实现复杂功能。 测试通过环境: windows x64 anaconda3python3.8 ultralytics8.3.81 flask1.1.2…

突破连接边界!O9201PM Wi-Fi 6 + 蓝牙 5.4 模块重新定义笔记本无线体验

在当今数字化时代,笔记本电脑已成为人们工作、学习和娱乐的必备工具。而无线连接技术,作为笔记本电脑与外界交互的关键桥梁,其性能的优劣直接关乎用户体验的好坏。当下,笔记本电脑无线连接领域存在诸多痛点,严重影响着…

2025 香港 Web3 嘉年华:全球 Web3 生态的年度盛会

自 2023 年首届香港 Web3 嘉年华成功举办以来,这一盛会已成为全球 Web3 领域规模最大、影响力最深远的行业活动之一。2025 年 4 月 6 日至 9 日,第三届香港 Web3 嘉年华将在香港盛大举行。本届活动由万向区块链实验室与 HashKey Group 联合主办、W3ME 承…

Windows11 新机开荒(二)电脑优化设置

目录 前言: 一、注册微软账号绑定权益 二、此电脑 桌面图标 三、系统分盘及默认存储位置更改 3.1 系统分盘 3.2 默认存储位置更改 四、精简任务栏 总结: 前言: 本文承接上一篇 新机开荒(一) 上一篇文章地址&…

[C++面试] 标准容器面试点

一、入门 1、vector和list的区别 [C面试] vector 面试点总结 vector 是动态数组,它将元素存储在连续的内存空间中。支持随机访问,即可以通过下标快速访问任意位置的元素,时间复杂度为 O(1),准确点是均摊O(1)。但在中间或开头插…

蓝桥杯每日一题

丢失的雨伞 题目思路代码演示 题目 今天晚上本来想练习一下前缀和与差分 结果给我搜出来这题(几乎没啥关系),我看半天有点思路但又下不了手哈哈,难受一批 在图书馆直接红温了 题目链接 思路 题目要求找到两个不重叠的区间&…

校园安全用电怎么保障?防触电装置来帮您

引言 随着教育设施的不断升级和校园用电需求的日益增长,校园电力系统的安全性和可靠性成为了学校管理的重要课题。三相智能安全配电装置作为一种电力管理设备,其在校园中的应用不仅能够提高电力系统的安全性,还能有效保障师生的用电安全&am…

Matlab 汽车二自由度转弯模型

1、内容简介 Matlab 187-汽车二自由度转弯模型 可以交流、咨询、答疑 2、内容说明 略 摘 要 本文前一部分提出了侧偏角和横摆角速度作为参数。描述了车辆运动的运动状态,其中文中使用的参考模型是二自由度汽车模型。汽车速度被认为是建立基于H.B.Pacejka的轮胎模…

OpenCV计算摄影学(20)非真实感渲染之增强图像的细节函数detailEnhance()

操作系统:ubuntu22.04 OpenCV版本:OpenCV4.9 IDE:Visual Studio Code 编程语言:C11 算法描述 此滤波器增强特定图像的细节。 cv::detailEnhance用于增强图像的细节,通过结合空间域和频率域的处理,提升图像中特定细节…

Java面试八股—Redis篇

一、Redis的使用场景 (一)缓存 1.Redis使用场景缓存 场景:缓存热点数据(如用户信息、商品详情),减少数据库访问压力,提升响应速度。 2.缓存穿透 正常的访问是:根据ID查询文章&…

2025-03-17 Unity 网络基础1——网络基本概念

文章目录 1 网络1.1 局域网1.2 以太网1.3 城域网1.4 广域网1.5 互联网(因特网)1.6 万维网1.7 小结 2 IP 地址2.1 IP 地址2.2 端口号2.3 Mac 地址2.4 小结 3 客户端与服务端3.1 客户端3.2 服务端3.3 网络游戏中的客户端与服务端 1 网络 ​ 在没有网络之前…