设计模式-迪米特法则(Law of Demeter, LoD)


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

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


原理详解

  1. 直接朋友的定义

    • 当前对象的成员变量。
    • 当前对象方法的参数。
    • 当前对象方法的返回值。
    • 当前对象方法中创建的对象(不推荐,但允许)。
  2. 禁止链式调用

    避免出现 a.getB().getC().doSomething() 的调用形式,这种"火车残骸式"代码会增加耦合性。

  3. 目标

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

应用案例

场景1:订单系统获取用户配送地址
错误设计(违反迪米特法则)
java 复制代码
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 的结构会影响客户端代码。
正确设计(遵循迪米特法则)
java 复制代码
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:文件系统目录结构遍历
错误设计(违反迪米特法则)
java 复制代码
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); // 递归调用暴露内部结构
    }
}

问题

  • 客户端需要了解目录的递归结构,耦合度高。
正确设计(遵循迪米特法则)
java 复制代码
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()
暴露集合内部结构 返回不可变集合或迭代器,避免直接操作
方法参数传递复杂对象 拆分参数为基本类型或接口

总结

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

相关推荐
charlie1145141911 小时前
从C++编程入手设计模式——装饰器模式
c++·设计模式·装饰器模式
缘友一世3 小时前
设计模式之五大设计原则(SOLID原则)浅谈
java·算法·设计模式
吴八月6 小时前
设计模式:简单工厂、工厂方法与抽象工厂
设计模式·简单工厂·抽象工厂·工厂方法
Your易元9 小时前
设计模式-状态模式
java·前端·算法·设计模式
杰_happy16 小时前
设计模式:原型模式(C++)
c++·设计模式·原型模式
OpenC++16 小时前
【C++】简单工厂模式/工厂方法模式/抽象工厂模式对比
c++·设计模式·简单工厂模式·工厂方法模式·抽象工厂模式
99乘法口诀万物皆可变20 小时前
C#设计模式之AbstractFactory_抽象工厂_对象创建新模式-学习
设计模式·c#·抽象工厂模式
英杰.王21 小时前
设计模式-接口隔离原则(Interface Segregation Principle, ISP)
设计模式·接口隔离原则
易元1 天前
设计模式-状态模式
后端·设计模式
英杰.王1 天前
设计模式-里氏替换原则(Liskov Substitution Principle, LSP)
设计模式·里氏替换原则