一文搞懂迪米特原则

迪米特原则,又称最少知识原则(Least Knowledge Principle, LKP),是面向对象设计和编程中的一项重要原则,其目标是降低类之间的耦合度,提高代码的可读性和可维护性。基本原则可以概括为:

原则描述:

  • 一个对象应当对其它对象有尽可能少的了解,即一个对象不应该直接访问非直接相关对象的方法或属性,而应通过其"朋友"或者代理进行间接交互。

  • "只和直接的朋友交谈",这里的"朋友"指的是:

    • 对象本身;
    • 对象所包含的对象(成员变量);
    • 对象的方法的参数;
    • 方法返回的对象。

具体实践方式:

  1. 限制对外部对象的直接引用:避免在一个类中直接访问另一个类的内部细节,例如不要直接操作外部对象的私有属性或方法。
  2. 通过委托进行通信:如果A对象需要调用B对象的某些功能,但A不是B的直接朋友,则可以引入一个中间者C,让C成为A的朋友并持有对B的引用,然后由C转发A对B的请求。
  3. 接口限定交互:使用接口而非具体实现来定义交互,这样客户端只需要知道接口而不是底层的具体实现,进一步减少耦合。

具体例子:

假设我们有一个简单的场景,其中包括学生(Student)、课程(Course)以及教师(Teacher)这三个类。按照迪米特原则,学生不应该直接操作课程或教师的相关细节,而是通过某种中介机制间接交互。

typescript 复制代码
// 定义抽象的接口以便解耦
interface ITeacher {
    getName(): string;
  }
  
  class Teacher implements ITeacher {
    private name: string;
  
    constructor(name: string) {
      this.name = name;
    }
  
    public getName() {
      return this.name;
    }
  }
  
  interface ICourse {
    getTeacher(): ITeacher;
    getName(): string;
  }
  
  class Course implements ICourse {
    private teacher: ITeacher;
    private courseName: string;
  
    constructor(courseName: string, teacher: ITeacher) {
      this.courseName = courseName;
      this.teacher = teacher;
    }
  
    public getTeacher() {
      return this.teacher;
    }
  
    public getName() {
      return this.courseName;
    }
  }
  
  class Student {
    private enrolledCourses: ICourse[];
  
    constructor(courses: ICourse[]) {
      this.enrolledCourses = courses;
    }
  
    // 遵循迪米特原则,学生不直接获取老师的信息,而是通过课程获取
    public getCourseTeacher(courseName: string): string | undefined {
      const course = this.enrolledCourses.find((course) => course.getName() === courseName);
      if (course) {
        const teacher = course.getTeacher();
        return teacher.getName(); // 间接获取教师的名字
      }
      return undefined;
    }
  }

在这个例子中:

  • 学生类(Student)只知道它所注册的课程(ICourse),并且通过课程类来获取教师的名字,而不会直接操作教师类(Teacher)。
  • 课程类(Course)提供了获取关联教师的方法,但它并不关心具体的教师实现,只需教师符合ITeacher接口即可。
  • 教师类(Teacher)隐藏了其内部实现细节,仅通过接口暴露名字。

因此,通过这样的设计,各层之间的耦合度得以降低,实现了迪米特原则。

优点:

  • 降低耦合度:通过减少类之间的直接联系,降低了修改一个类可能带来的连锁反应,提高了系统的稳定性和可维护性。
  • 增强模块独立性:遵循迪米特原则的设计可以使各个模块更加独立,每个模块仅需关注自己的职责和与其紧密相关的部分,更容易理解和测试。
  • 提高复用性和扩展性:因为类间的依赖减少了,所以更容易替换组件或增加新的功能,而不需要对现有大量代码进行修改。

潜在挑战:

  • 过度中介化:严格遵守迪米特原则可能会导致系统中出现大量中介类,这些类仅仅为了传递调用关系而存在,这可能导致系统变得复杂且难以理解。
  • 权衡:在实际设计中需要适度平衡迪米特原则和其他设计原则,确保系统的简洁性和性能不受影响。

总之,迪米特原则是一种指导设计思路的原则,帮助开发者组织代码结构,促进良好的封装性,并努力减少因类间过度耦合而导致的问题。在实践中,灵活运用这一原则能够改善软件设计的整体质量。

相关推荐
辻戋1 小时前
从零实现React Scheduler调度器
前端·react.js·前端框架
徐同保1 小时前
使用yarn@4.6.0装包,项目是react+vite搭建的,项目无法启动,报错:
前端·react.js·前端框架
Qrun2 小时前
Windows11安装nvm管理node多版本
前端·vscode·react.js·ajax·npm·html5
中国lanwp2 小时前
全局 npm config 与多环境配置
前端·npm·node.js
JELEE.3 小时前
Django登录注册完整代码(图片、邮箱验证、加密)
前端·javascript·后端·python·django·bootstrap·jquery
TeleostNaCl5 小时前
解决 Chrome 无法访问网页但无痕模式下可以访问该网页 的问题
前端·网络·chrome·windows·经验分享
前端大卫7 小时前
为什么 React 中的 key 不能用索引?
前端
你的人类朋友7 小时前
【Node】手动归还主线程控制权:解决 Node.js 阻塞的一个思路
前端·后端·node.js
小李小李不讲道理9 小时前
「Ant Design 组件库探索」五:Tabs组件
前端·react.js·ant design
毕设十刻9 小时前
基于Vue的学分预警系统98k51(程序 + 源码 + 数据库 + 调试部署 + 开发环境配置),配套论文文档字数达万字以上,文末可获取,系统界面展示置于文末
前端·数据库·vue.js