设计模式-依赖倒置原则(3)

设计原则

单一职责原则
里氏替换原则
依赖倒置原则
接口隔离原则
迪米特法则
开闭原则

依赖倒置原则的定义

高层模块不应该依赖低层模块, 两者都应该依赖其抽象;
抽象不应该依赖细节;
细节应该依赖抽象。

高层模块和低层模块容易理解, 每一个逻辑的实现都是由原子逻辑组成的, 不可分割的原子逻辑就是低层模块, 原子逻辑的再组装就是高层模块。 那什么是抽象? 什么又是细节呢? 在Java语言中, 抽象就是指接口或抽象类, 两者都是不能直接被实例化的; 细节就是实现类, 实现接口或继承抽象类而产生的类就是细节, 其特点就是可以直接被实例化, 也就是可以加上一个关键字new产生一个对象。
依赖倒置原则在Java语言中的表现就是:

模块间的依赖通过抽象发生, 实现类之间不发生直接的依赖关系, 其依赖关系是通过接口或抽象类产生的;
接口或抽象类不依赖于实现类;
实现类依赖接口或抽象类。

看1个反例不适用依赖倒置

设计模式-依赖倒置原则(3)_第1张图片

上述代码违背“细节应该依赖抽象”的原则,上述代码很容易实现司机开奔驰,但是如果司机要开宝马该怎么办?麻烦就来了!

先增加1个宝马类

public class BMW {
  public void run(){
    System.out.println("宝马汽车开始运行...");
  }
}

宝马车也产生了, 但是我们却没有办法让张三开动起来, 为什么? 张三没有开动宝马车的方法呀!此时Driver也需要修改,并且增加1个新方法才能开宝马,需要修改2个地方。说明设计出现了问题:司机类和奔驰车类之间是紧耦合的关系, 其导致的结果就是系统的可维护性大大降低,可读性降低。
使用依赖倒置的原则修改代码后如下图:

设计模式-依赖倒置原则(3)_第2张图片

贯彻“抽象不应该依赖细节”, 也就是我们认为抽象(ICar接口) 不依赖BMW和Benz两个实现类(细节) , 因此在高层次的模块中应用都是抽象,Client的实现过程:

public class Client {
    public static void main(String[] args) {
      IDriver zhangSan = new Driver();
      ICar benz = new Benz();
      //张三开奔驰车
      zhangSan.drive(benz);
    }
}

Client属于高层业务逻辑, 它对低层模块的依赖都建立在抽象上, 如果张三要开宝马车, 也很容易, 我们只要修改业务场景类就可以:

public class Client {
    public static void main(String[] args) {
        IDriver zhangSan = new Driver();
        ICar bmw = new BMW();
        //张三开奔驰车
        zhangSan.drive(bmw);
    }
}

在新增加低层模块时, 只修改了业务场景类, 也就是高层模块, 对其他低层模块如Driver类不需要做任何修改, 业务就可以运行, 把“变更”引起的风险扩散降到最低。

依赖关系的三种写法

1.构造函数传递依赖对象

public interface IDriver {
   //是司机就应该会驾驶汽车
   public void drive();
}

public class Driver implements IDriver{
   private ICar car;
        //构造函数注入
        public Driver(ICar _car){
        this.car = _car;
    }
    //司机的主要职责就是驾驶汽车
    public void drive(){
        this.car.run();
    }
}

2.Setter方法传递依赖对象

public interface IDriver {
    //车辆型号
    public void setCar(ICar car);
    public void drive();
}

public class Driver implements IDriver{
    private ICar car;
    public void setCar(ICar car){
        this.car = car;
    }
    //司机的主要职责就是驾驶汽车
    public void drive(){
        this.car.run();
    }
}

3.接口声明依赖对象

public interface IDriver {
    //是司机就应该会驾驶汽车
    public void drive(ICar car);
}

总结

依赖倒置原则的本质就是通过抽象(接口或抽象类) 使各个类或模块的实现彼此独立,不互相影响, 实现模块间的松耦合, 我们怎么在项目中使用这个规则呢? 只要遵循以下的几个规则就可以:
 

每个类尽量都有接口或抽象类, 或者抽象类和接口两者都具备
变量的表面类型尽量是接口或者是抽象类
任何类都不应该从具体类派生
尽量不要覆写基类的方法
结合里氏替换原则使用

依赖倒置原则是6个设计原则中最难以实现的原则, 它是实现开闭原则的重要途径, 依赖倒置原则没有实现, 就别想实现对扩展开放, 对修改关闭。 在项目中, 大家只要记住是“面向接口编程”就基本上抓住了依赖倒置原则的核心。

你可能感兴趣的:(设计模式)