1. 概念
- 迪米特法则(Demeter Principle)又叫最少知道原则,即一个类对自己依赖的类知道的越少越好。
- 迪米特法则还有个更简单的定义:只与直接的朋友通信
- 朋友:有耦合(依赖,关联,组合,聚合)关系的对象
- 直接的朋友:成员变量,方法参数,方法返回值中的类
2. 优点
- 降低耦合性,提高模块功能的独立性
- 非常实用,能直接用于开发环境
3. 代码示例
示例1(违背迪米特法则的案例)
package com.bz.design.principle.demeter;
import java.util.ArrayList;
import java.util.List;
// 客户端
public class Demeter001 {
public static void main(String[] args) {
//创建了一个SchoolManager对象
SchoolManager schoolManager = new SchoolManager();
//输出学院的员工id和学校总部的员工信息
schoolManager.printAllEmployee(new CollegeManager());
}
}
// 总部员工
class Employee {
private String id;
public void setId(String id) {
this.id = id;
}
public String getId() {
return this.id;
}
}
// 学院员工
class CollegeEmployee {
private String id;
public void setId(String id) {
this.id = id;
}
public String getId() {
return this.id;
}
}
// 管理学院员工的类
class CollegeManager {
// 返回学院的所有员工
public List<CollegeEmployee> getAllEmployee() {
List<CollegeEmployee> list = new ArrayList<>();
for (int i = 0; i < 10; i++) {
CollegeEmployee emp = new CollegeEmployee();
emp.setId("学院员工id=" + i);
list.add(emp);
}
return list;
}
}
// 学校管理类
// 分析 SchoolManager 类的直接朋友 Employee、CollegeManager
// CollegeEmployee 不是直接朋友,违反了迪米特法则
class SchoolManager {
// 返回学校总部的员工
public List<Employee> getAllEmployee() {
List<Employee> list = new ArrayList<>();
for (int i = 0; i < 5; i++) {
Employee emp = new Employee();
emp.setId("学校总部员工id=" + i);
list.add(emp);
}
return list;
}
// 该方法完成输出学校总部和学院员工信息的方法
void printAllEmployee(CollegeManager sub) {
// 分析问题
// 1.这里的CollegeEmployee 不是SchoolManager的直接朋友
// 2.CollegeEmployee 是以局部变量方式出现在SchoolManager
// 3.违反了迪米特法则
List<CollegeEmployee> list1 = sub.getAllEmployee();
System.out.println("----------学院员工----------");
for (CollegeEmployee e : list1) {
System.out.println(e.getId());
}
List<Employee> list2 = this.getAllEmployee();
System.out.println("------------学校总部员工------------");
for (Employee e : list2) {
System.out.println(e.getId());
}
}
}
示例2
- 应用实例改进
- 前面设计的问题在于SchoolManager中,CollegeEmployee类并不不是SchoolManager类的直接朋友(分析)
- 按照迪米特法则,应该避免类中出现这样非直接朋友关系的耦合
- 对代码按照迪米特法则进行改进
package com.bz.design.principle.demeter.improve;
import java.util.ArrayList;
import java.util.List;
/**
* 应用实例改进
* 1)前面设计的问题在于SchoolManager中,CollegeEmployee类并不不是SchoolManager类的直接朋友(分析)
* 2)按照迪米特法则,应该避免类中出现这样非直接朋友关系的耦合
* 3)对代码按照迪米特法则进行改进
*/
public class DemeterImprove {
public static void main(String[] args) {
System.out.println("---使用迪米特法则改进---");
//创建了一个SchoolManager对象
SchoolManager schoolManager = new SchoolManager();
//输出学院的员工id和学校总部的员工信息
schoolManager.printAllEmployee(new CollegeManager());
}
}
// 总部员工
class Employee {
private String id;
public void setId(String id) {
this.id = id;
}
public String getId() {
return this.id;
}
}
// 学院员工
class CollegeEmployee {
private String id;
public void setId(String id) {
this.id = id;
}
public String getId() {
return this.id;
}
}
// 管理学院员工的类
class CollegeManager {
// 返回学院的所有员工
public List<CollegeEmployee> getAllEmployee() {
List<CollegeEmployee> list = new ArrayList<>();
for (int i = 0; i < 10; i++) {
CollegeEmployee emp = new CollegeEmployee();
emp.setId("学院员工id=" + i);
list.add(emp);
}
return list;
}
// 改进点
// 输出学院员工的信息
public void printEmployee() {
// 获取到学院员工
List<CollegeEmployee> list1 = this.getAllEmployee();
System.out.println("----------学院员工----------");
for (CollegeEmployee e : list1) {
System.out.println(e.getId());
}
}
}
// 学校管理类
class SchoolManager {
// 返回学校总部的员工
public List<Employee> getAllEmployee() {
List<Employee> list = new ArrayList<>();
for (int i = 0; i < 5; i++) {
Employee emp = new Employee();
emp.setId("学校总部员工id=" + i);
list.add(emp);
}
return list;
}
// 该方法完成输出学校总部和学院员工信息的方法
void printAllEmployee(CollegeManager sub) {
// 分析问题
// 将输出学院的员工方法,封装到CollegeManager
sub.printEmployee();
List<Employee> list2 = this.getAllEmployee();
System.out.println("------------学校总部员工------------");
for (Employee e : list2) {
System.out.println(e.getId());
}
}
}
4. 应用注意事项
-
不要干
- objectA.getObjectB().doSomething();
- objectA.getObjectB().getObjectC().doSomething();
-
遵循迪米特法则的设计模式
- 注意事项