一、迪米特法则介绍
迪米特法则(Demeter Principle)又叫最少知道原则。它是指一个类对自己依赖的类知道得越少越好。也就是说,对于被依赖的类,不管多么复杂,都尽量将逻辑封装在类的内部,对外除了提供public方法外,不对外泄露任何消息。
迪米特法则还有一个更简单的定义:只与直接朋友通信。这里的直接朋友的定义是:每个对象都会与其他对象有耦合关系,只要两个对象之间有耦合关系,我们就说这两个对象之间是朋友关系。耦合的方式很多,如依赖、关联、组合、聚合等。其中,我们称出现成员变量、方法参数、方法返回值中的类为直接的朋友,而出现在局部变量中的类不是直接朋友,也就是说,陌生的类最好不要以局部变量的形式出现在类的内部。
二、案例讲解
有一个学校,下属有各个学院和总部,现要求打印出学校总部员工ID和学院员工ID。代码如下:
//学校总部员工类
public class Employee{
private String id;
public String getId(){
return id;
}
public void setId(String id){
this.id = id;
}
}
//学院的员工类
public class CollegeEmployee{
private String id;
public String getId(){
return id;
}
public void setId(String id){
this.id = id;
}
}
//管理学院员工的管理类
public class CollegeManager{
public List<CollegeEmployee> listCollegeEmployee(){
public List<CollegeEmployee> list = new ArrayList<>();
for(int = 0; i < 10; i++){
CollegeEmployee e = new CollegeEmployee();
e.setId("学院员工id:" + id);
list.add(e);
}
return list;
}
}
//学校管理类
public class SchoolManager{
//返回总部所有员工
public List<Employee> listEmployee(){
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 printEmployee(CollegeManager sub){
List<CollegeEmployee> collegeEmpList = sub.listCollegeEmployee();
System.out.println("------------学院员工----------------");
for(CollegeEmployee e collegeEmpList){
System.out.println(e.getId);
}
List<Employee> empList = this.listEmployee();
System.out.println("------------学校总部员工----------------");
for(Employee e empList){
System.out.println(e.getId);
}
}
}
我们分析类SchoolManager,根据直接朋友的定义可知,类Employee和类CollegeManager是直接朋友,类Employee是以方法的返回值作为直接朋友,而类CollegeManager是以方法的参数作为直接。而类CollegeEmployee不是SchoolManager的直接朋友,因为CollegeEmployee是以局部变量方式出现在SchoolManager,违反了迪米特法则。
按照迪米特法法,应该避免类中出现这样非直接朋友关系的耦合,对上面的SchoolManager、CollegeManager类进行改进。如下:
//管理学院员工的管理类
public class CollegeManager{
public List<CollegeEmployee> listCollegeEmployee(){
public List<CollegeEmployee> list = new ArrayList<>();
for(int = 0; i < 10; i++){
CollegeEmployee e = new CollegeEmployee();
e.setId("学院员工id:" + id);
list.add(e);
}
return list;
}
public void printCollegeEmp(){
List<CollegeEmployee> collegeEmpList = this.listCollegeEmployee();
System.out.println("------------学院员工----------------");
for(CollegeEmployee e collegeEmpList){
System.out.println(e.getId);
}
}
}
//学校管理类
public class SchoolManager{
//返回总部所有员工
public List<Employee> listEmployee(){
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 printEmployee(CollegeManager sub){
sub.listCollegeEmployee();
List<Employee> empList = this.listEmployee();
System.out.println("------------学校总部员工----------------");
for(Employee e empList){
System.out.println(e.getId);
}
}
}
三、迪米特法则注意事项和细节
1)迪米特法则的核心是降低类之间的耦合。一个对象应该对其他对象保存最少的了解,类与类关系越密切,耦合度越高
2)但是注意:由于每个类都减少了不必要的依赖,因此迪米特法则只是要求降低类间(对象间)耦合关系,并不是要求完成没有依赖关系