设计模式-迪米特法则(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()
暴露集合内部结构 返回不可变集合或迭代器,避免直接操作
方法参数传递复杂对象 拆分参数为基本类型或接口

总结

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

相关推荐
我登哥MVP16 小时前
Spring Boot 从“会用”到“精通”:请求映射原理
java·spring boot·后端·spring·servlet·maven·intellij-idea
Sam_Deep_Thinking16 小时前
结算分摊的策略模式:不同营销活动的扣点计算方案
java·设计模式·架构·系统架构
故渊at19 小时前
系列一:架构思想进阶 | 第3篇 SOLID 原则与设计模式实战:从“代码搬运工”到“架构师”的必经之路
观察者模式·设计模式·重构·架构·代理模式
我登哥MVP19 小时前
Spring Boot 从“会用”到“精通”:Converter 原理
java·spring boot·servlet·maven·mybatis·converter
老码观察1 天前
设计模式实战解读(十一):外观模式——给复杂系统套一层壳
python·设计模式·外观模式
AI大法师2 天前
奥迪 AUDI 案例:母品牌和新业务怎么拆?
大数据·设计模式·汽车
bryant_meng2 天前
【Design Patterns】23 Design Patterns: The Ultimate Developer‘s Toolkit
设计模式·编程·计算机科学·设计·工程
狂人开飞机2 天前
18. 中介者模式(Mediator Pattern)
设计模式·c#·中介者模式
咖啡八杯2 天前
GoF设计模式——外观模式
java·设计模式·外观模式
江湖中的阿龙2 天前
23种设计模式
java·开发语言·设计模式