当前位置: 首页 > news >正文

【设计模式-迪米特法则】

迪米特法则(Law of Demeter,LoD),也称为最少知识原则(Principle of Least Knowledge),是一种面向对象编程中的设计原则。它的核心思想是:一个对象应当尽可能少地了解其他对象,即只与直接相关的对象通信,而不要过度依赖外部对象的内部细节。通过减少对象之间的耦合度,提升代码的可维护性、可扩展性以及模块化程度。

1. 迪米特法则的定义

迪米特法则的定义可以简单概括为:

  • 不要与陌生人说话:一个对象不应当调用其他对象的内部方法(除了其直接依赖的对象)。
  • 对象只与其直接关联的对象交互,避免与依赖链上的其他对象产生直接通信。

2. 具体规则

根据迪米特法则,一个对象的方法只能调用以下几类对象的方法:

  • 自身对象:即当前对象的方法。
  • 其成员对象:对象的成员变量、属性(包括集合中的元素)。
  • 方法参数:当前方法中传入的参数对象。
  • 方法创建的局部对象:当前方法中创建的局部变量对象。
  • 全局变量(如果存在)。

这个原则的目的是减少不同类之间的耦合,使得一个模块或类不需要过多地了解另一个模块的内部实现,进而降低系统中各个模块之间的依赖性。

3. 迪米特法则的目的

  • 降低耦合:迪米特法则通过减少对象之间的相互依赖,避免由于一个类的修改而影响到其他类,增强系统的灵活性和可维护性。
  • 提高内聚性:遵守迪米特法则,可以使类的职责更加明确,每个类只处理自己的相关操作,增强类的内聚性。
  • 增强可读性:减少类与类之间的直接通信后,代码的整体结构更加清晰,使得代码的可读性和可理解性提高。

4. 违反迪米特法则的典型例子

迪米特法则建议避免“链式调用”或过度深入依赖对象的内部结构。假设有一个汽车类和发动机类的设计,以下是一个违反迪米特法则的示例:

class Engine {public String getFuelType() {return "Gasoline";}
}class Car {private Engine engine = new Engine();public Engine getEngine() {return engine;}
}public class Main {public static void main(String[] args) {Car car = new Car();// 通过Car对象获取Engine对象,并进一步调用Engine的getFuelType()方法String fuelType = car.getEngine().getFuelType();System.out.println("Fuel type: " + fuelType);}
}
违背迪米特法则的地方:

在上面的代码中,Main 类不仅与 Car 类产生了依赖,还与 Engine 类产生了依赖。这意味着如果 Engine 类的实现发生变化,Main 类也可能需要修改。这种情况增加了类之间的耦合性,违背了迪米特法则。

5. 符合迪米特法则的设计

为了遵守迪米特法则,Car 类应该封装其内部的 Engine 细节,只提供获取燃料类型的方法,而不暴露 Engine 对象的内部结构:

class Engine {public String getFuelType() {return "Gasoline";}
}class Car {private Engine engine = new Engine();// 提供获取燃料类型的方法,而不是暴露Engine对象public String getFuelType() {return engine.getFuelType();}
}public class Main {public static void main(String[] args) {Car car = new Car();// 直接通过Car对象获取燃料类型,不需要访问Engine对象String fuelType = car.getFuelType();System.out.println("Fuel type: " + fuelType);}
}
改进的设计:
  • Main 类现在只和 Car 类打交道,并不需要了解 Engine 类的存在。这减少了类之间的耦合度,符合迪米特法则的要求。
  • Car 类承担了管理 Engine 类的责任,封装了获取 Engine 的内部逻辑,外部不需要关心 Engine 的具体实现。

6. 迪米特法则的好处

  • 提高模块独立性:每个模块或类的实现细节不直接暴露给其他模块,因此更容易进行独立开发和测试。
  • 减少修改的影响范围:当一个类的实现细节发生变化时,由于其他类不依赖于这些细节,因此不需要修改其他类的代码。
  • 增强代码的可维护性:对象之间的依赖关系少了,修改某个对象时,不会影响到太多的其他对象,从而提高了系统的可维护性。

7. 迪米特法则的应用场景

迪米特法则在以下场景中非常适用:

  • 大型复杂系统:在设计大型复杂系统时,减少对象之间的耦合可以避免系统结构的混乱,增强系统的可扩展性。
  • 面向对象设计中:在进行面向对象编程时,迪米特法则是一项重要的设计原则,帮助开发者避免创建过于复杂的依赖关系。
  • 模块化开发:模块之间的通信应尽量简单,避免过度依赖其他模块的细节,保持各个模块的独立性。

8. 迪米特法则的缺点

虽然迪米特法则有诸多优点,但在某些情况下也可能带来一些问题:

  • 过度封装:为了严格遵守迪米特法则,开发者可能会过度封装一些细节,导致代码复杂性增加,增加类的职责,从而违背单一职责原则。
  • 性能问题:在某些场景中,为了隐藏内部结构和减少直接依赖,可能引入额外的方法调用,导致系统性能下降。

9. 总结

迪米特法则强调降低对象之间的耦合,通过减少类之间的直接依赖,提升代码的可维护性、可扩展性以及可复用性。在实践中,遵守迪米特法则有助于设计高内聚、低耦合的系统,使得系统中的各个模块更加独立、灵活。然而,迪米特法则并不是绝对的,开发者在应用时应当权衡封装与性能、复杂度之间的关系,避免过度封装导致代码臃肿或职责不清。

相关文章:

【设计模式-迪米特法则】

迪米特法则(Law of Demeter,LoD),也称为最少知识原则(Principle of Least Knowledge),是一种面向对象编程中的设计原则。它的核心思想是:一个对象应当尽可能少地了解其他对象&#x…...

Webpack安装

全局安装 npm install -g webpack webpack-cli安装后查看版本号: webpack -v初始化项目 npm init -yJS打包 webpack目录下创建配置文件webpack.config.js 以下配置的意思是:读取当前项目目录下src文件夹中的main.js(入口文件&#xff09…...

前端开发学习(一)VUE框架概述

一、MVC模式与MVVM模式 1.1mvc模式 MVC模式是移动端应用广泛的软件架构之一,MVC模式将应用程序划分为3部分:Model(数据模型)、View(用户界面视图)和Controller(控制器)。MVC模式的执行过程是将View层展示给用户,也就是通过 HTML页面接受用户动作&#…...

Linux操作系统的背景、发展历程及对比分析

1. UNIX发展历史 unix_百度百科 UNIX操作系统作为现代操作系统的奠基石,其发展历史可以追溯到20世纪60年代末。1969年,贝尔实验室的Ken Thompson、Dennis Ritchie以及他们的同事们为了实现一种多任务的、可移植的、简洁而高效的操作系统,开发…...

gaussdb 基础管理 数据库 表 用户 模式 权限 存储过程

数据库database #创建数据库,指定字符集UTF8,缺省情况下新数据库将通过复制标准系统数据库template0来创建,且仅支持使用template0来创建。 CREATE DATABASE devdb ENCODING UTF8 template template0; CREATE DATABASE testdb; 标识符的命名…...

i9-11900H+3070laptop+win10下的yolov5配置

参考博客:https://blog.csdn.net/qq_67105081/article/details/138232424 关键问题: 1、由之前装的CUDA11.6(有篇博客上可以换版本)CUDNN8.4.0Python3.9.13推后在指定的虚拟环境中装了Pytorch1.12.1(在AnnacondaPrompt下用pip命令…...

SpringBoot日常:封装redission starter组件

文章目录 逻辑实现POM.xmlRedissionConfigRedissionPropertiesRedissionUtilsspring.factories 功能测试application.yml配置POM.xmlTestController运行测试 本章内容主要介绍如何通过封装相关的redission连接配置和工具类,最终完成一个通用的redission starter。并…...

腾讯云技术深度解析:构建高效云原生应用与数据安全管理

腾讯云技术深度解析:构建高效云原生应用与数据安全管理 在当今快速发展的技术环境中,云计算已经成为企业数字化转型的关键驱动力。腾讯云作为中国领先的云服务提供商,凭借其卓越的技术和创新能力,为企业提供了高效、可扩展的云原…...

ACM与蓝桥杯竞赛指南 基本输入输出格式二

A B || 继续看第二个AB问题,A B || ,大家可以先自行读题,或者有经验直接看输入输出格式,发现依然是求AB,但它的输入数据为: 2 1 5 10 20 输出: 6 20 此题相比第一道而言,本…...

解决SolidWorks装配体无法更改透明度问题

这个问题是在零件上各个部件显示正常,且透明度可以更改,但是一到装配体上就出现问题都变成了灰色。更改透明度也不行。 解决方法: 1、因为该装配体里面存在过多的零部件层级的自定义外观(这些外观可能互相之前有了干扰&#xff0…...

2024_newstar_week1_crypto

baby_mod 题目 from Crypto.Util.number import * from enc import flagm bytes_to_long(flag) p getPrime(512) q getPrime(512) r getPrime(777) t getPrime(777) tmp getPrime(15) e 65537 n p*q print(f"c {pow(m,e,n)}") print(f"leak {p*r-q*…...

6.2 URDF集成Rviz基本流程

前面介绍过,URDF 不能单独使用,需要结合 Rviz 或 Gazebo,URDF 只是一个文件,需要在 Rviz 或 Gazebo 中渲染成图形化的机器人模型,当前,首先演示URDF与Rviz的集成使用,因为URDF与Rviz的集成较之于…...

双系统一体机电脑无法启动报错“Something has gone serously wrong: SBAT self-check failed: Security Policy Violation”

双系统一体机电脑无法启动 问题搜索解决办法解决开启时 F2 进入系统设置界面选择“疑难解答”选择“高级选项”选择“UEFI固件设置”进入“Start Menu”界面选择“Security”关闭相关选项 问题 在2024/8/14日Windows环境,系统更新了两个Windows更新项后&#xff0c…...

八股面试2(自用)

mysql存储引擎 存储引擎:定义数据的存储方式,以及数据读取的实现逻辑 在以前数据库5.5默认MyISAM引擎,之后默认InnoDB引擎 MyISAM引擎的数据和索引是分开存储的,InnoDb将索引和文件存储在同一个文件。 MyISAM不支持事务&#…...

Leetcode 347 Top K Frequent Elements

题意: 求前k个出现频率最高的元素 首先得到一个频率图这是肯定的,下一步要考虑建立一个堆,堆中保存着前k个频率最大的数字,这个怎么做,可以用customized cmp来做,把数字存进去完事儿。注意这里不用 保存所有…...

[Linux网络编程]03-TCP协议

一.TCP协议数据通信的过程 TCP数据报如下,数据报中的标志位双端通信的关键。 三次握手: 1.客户端向服务端发送SYN标志位,请求建立连接,同时发送空包 2.服务端向客户端回发ACK标志位(即确认标志位,任何一端发送数据后都需要另一端…...

Windows和Linux在客户端/服务端在安全攻防方面的区别

Windows和Linux在客户端/服务端的安全攻防上存在一些显著区别,主要体现在系统架构、权限管理、安全工具、更新机制以及社区支持等方面。 一、系统架构与设计差异 1. 内核架构 Windows:Windows是一个闭源的操作系统,由微软开发和维护&#…...

VUE 仿神州租车-开放平台

项目背景: 神州租车是一家提供汽车租赁服务的公司,其API开放平台为开发者提供了访问神州租车相关服务和数据的接口。用VUE技术来仿照其开发平台。 成果展示: 首页: API文档: 关于我们:...

计算机的错误计算(一百二十九)

摘要 用错数解释计算机的错误计算(一百二十七)中的计算错误的原因。 从(一百二十七)知, 有四种形式: 四个 分别有1位、8位、8位、0位错误数字。 下面用错数解释前面3个错误计算的由来。 (1&a…...

process.platform 作用

process.platform 可以获取当前运行 Node.js 进程的操作系统平台的信息。 一、平台特定的代码执行 1. 适应不同操作系统 不同的操作系统可能有不同的行为、文件路径格式、命令行参数等。通过检查process.platform的值,可以根据当前运行的平台执行特定的代码逻辑。…...

Docker 离线安装指南

参考文章 1、确认操作系统类型及内核版本 Docker依赖于Linux内核的一些特性,不同版本的Docker对内核版本有不同要求。例如,Docker 17.06及之后的版本通常需要Linux内核3.10及以上版本,Docker17.09及更高版本对应Linux内核4.9.x及更高版本。…...

Spark 之 入门讲解详细版(1)

1、简介 1.1 Spark简介 Spark是加州大学伯克利分校AMP实验室(Algorithms, Machines, and People Lab)开发通用内存并行计算框架。Spark在2013年6月进入Apache成为孵化项目,8个月后成为Apache顶级项目,速度之快足见过人之处&…...

AI Agent与Agentic AI:原理、应用、挑战与未来展望

文章目录 一、引言二、AI Agent与Agentic AI的兴起2.1 技术契机与生态成熟2.2 Agent的定义与特征2.3 Agent的发展历程 三、AI Agent的核心技术栈解密3.1 感知模块代码示例:使用Python和OpenCV进行图像识别 3.2 认知与决策模块代码示例:使用OpenAI GPT-3进…...

练习(含atoi的模拟实现,自定义类型等练习)

一、结构体大小的计算及位段 (结构体大小计算及位段 详解请看:自定义类型:结构体进阶-CSDN博客) 1.在32位系统环境,编译选项为4字节对齐,那么sizeof(A)和sizeof(B)是多少? #pragma pack(4)st…...

通过Wrangler CLI在worker中创建数据库和表

官方使用文档:Getting started Cloudflare D1 docs 创建数据库 在命令行中执行完成之后,会在本地和远程创建数据库: npx wranglerlatest d1 create prod-d1-tutorial 在cf中就可以看到数据库: 现在,您的Cloudfla…...

相机Camera日志分析之三十一:高通Camx HAL十种流程基础分析关键字汇总(后续持续更新中)

【关注我,后续持续新增专题博文,谢谢!!!】 上一篇我们讲了:有对最普通的场景进行各个日志注释讲解,但相机场景太多,日志差异也巨大。后面将展示各种场景下的日志。 通过notepad++打开场景下的日志,通过下列分类关键字搜索,即可清晰的分析不同场景的相机运行流程差异…...

日常一水C

多态 言简意赅:就是一个对象面对同一事件时做出的不同反应 而之前的继承中说过,当子类和父类的函数名相同时,会隐藏父类的同名函数转而调用子类的同名函数,如果要调用父类的同名函数,那么就需要对父类进行引用&#…...

c# 局部函数 定义、功能与示例

C# 局部函数:定义、功能与示例 1. 定义与功能 局部函数(Local Function)是嵌套在另一个方法内部的私有方法,仅在包含它的方法内可见。 • 作用:封装仅用于当前方法的逻辑,避免污染类作用域,提升…...

VisualXML全新升级 | 新增数据库编辑功能

VisualXML是一个功能强大的网络总线设计工具,专注于简化汽车电子系统中复杂的网络数据设计操作。它支持多种主流总线网络格式的数据编辑(如DBC、LDF、ARXML、HEX等),并能够基于Excel表格的方式生成和转换多种数据库文件。由此&…...

实战设计模式之模板方法模式

概述 模板方法模式定义了一个操作中的算法骨架,并将某些步骤延迟到子类中实现。模板方法使得子类可以在不改变算法结构的前提下,重新定义算法中的某些步骤。简单来说,就是在一个方法中定义了要执行的步骤顺序或算法框架,但允许子类…...