【设计模式-迪米特法则】
迪米特法则(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(入口文件)…...
前端开发学习(一)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、因为该装配体里面存在过多的零部件层级的自定义外观(这些外观可能互相之前有了干扰࿰…...
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更新项后,…...
八股面试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的值,可以根据当前运行的平台执行特定的代码逻辑。…...
业务系统对接大模型的基础方案:架构设计与关键步骤
业务系统对接大模型:架构设计与关键步骤 在当今数字化转型的浪潮中,大语言模型(LLM)已成为企业提升业务效率和创新能力的关键技术之一。将大模型集成到业务系统中,不仅可以优化用户体验,还能为业务决策提供…...
ubuntu搭建nfs服务centos挂载访问
在Ubuntu上设置NFS服务器 在Ubuntu上,你可以使用apt包管理器来安装NFS服务器。打开终端并运行: sudo apt update sudo apt install nfs-kernel-server创建共享目录 创建一个目录用于共享,例如/shared: sudo mkdir /shared sud…...
【入坑系列】TiDB 强制索引在不同库下不生效问题
文章目录 背景SQL 优化情况线上SQL运行情况分析怀疑1:执行计划绑定问题?尝试:SHOW WARNINGS 查看警告探索 TiDB 的 USE_INDEX 写法Hint 不生效问题排查解决参考背景 项目中使用 TiDB 数据库,并对 SQL 进行优化了,添加了强制索引。 UAT 环境已经生效,但 PROD 环境强制索…...
【Redis技术进阶之路】「原理分析系列开篇」分析客户端和服务端网络诵信交互实现(服务端执行命令请求的过程 - 初始化服务器)
服务端执行命令请求的过程 【专栏简介】【技术大纲】【专栏目标】【目标人群】1. Redis爱好者与社区成员2. 后端开发和系统架构师3. 计算机专业的本科生及研究生 初始化服务器1. 初始化服务器状态结构初始化RedisServer变量 2. 加载相关系统配置和用户配置参数定制化配置参数案…...
2.Vue编写一个app
1.src中重要的组成 1.1main.ts // 引入createApp用于创建应用 import { createApp } from "vue"; // 引用App根组件 import App from ./App.vue;createApp(App).mount(#app)1.2 App.vue 其中要写三种标签 <template> <!--html--> </template>…...
【配置 YOLOX 用于按目录分类的图片数据集】
现在的图标点选越来越多,如何一步解决,采用 YOLOX 目标检测模式则可以轻松解决 要在 YOLOX 中使用按目录分类的图片数据集(每个目录代表一个类别,目录下是该类别的所有图片),你需要进行以下配置步骤&#x…...
leetcodeSQL解题:3564. 季节性销售分析
leetcodeSQL解题:3564. 季节性销售分析 题目: 表:sales ---------------------- | Column Name | Type | ---------------------- | sale_id | int | | product_id | int | | sale_date | date | | quantity | int | | price | decimal | -…...
【RockeMQ】第2节|RocketMQ快速实战以及核⼼概念详解(二)
升级Dledger高可用集群 一、主从架构的不足与Dledger的定位 主从架构缺陷 数据备份依赖Slave节点,但无自动故障转移能力,Master宕机后需人工切换,期间消息可能无法读取。Slave仅存储数据,无法主动升级为Master响应请求ÿ…...
(一)单例模式
一、前言 单例模式属于六大创建型模式,即在软件设计过程中,主要关注创建对象的结果,并不关心创建对象的过程及细节。创建型设计模式将类对象的实例化过程进行抽象化接口设计,从而隐藏了类对象的实例是如何被创建的,封装了软件系统使用的具体对象类型。 六大创建型模式包括…...
如何应对敏捷转型中的团队阻力
应对敏捷转型中的团队阻力需要明确沟通敏捷转型目的、提升团队参与感、提供充分的培训与支持、逐步推进敏捷实践、建立清晰的奖励和反馈机制。其中,明确沟通敏捷转型目的尤为关键,团队成员只有清晰理解转型背后的原因和利益,才能降低对变化的…...
