设计模式七大原则 - 依赖倒转原则

依赖倒转原则

基本介绍

  1. 高层模块不应该依赖低层模块,二者都应该依赖其抽象
  2. 抽象不应该依赖细节,细节应该依赖抽象
  3. 依赖倒转的中心思想是面向接口编程
  4. 依赖倒转的原则是基于这样的设计理念:相对于细节的多变性,抽象的东西要稳定的多。以抽象为基础搭建的架构比以细节为基础搭建的架构要稳定的多。在java中,抽象指的是接口或抽象类,细节就是具体的实现类
  5. 使用接口或抽象类的目的是制定好规范,而不涉及任何的具体操作。把展现细节的任务交给他们的实现类去完成

注意的事项和细节

  1. 低层模块尽量都要有抽象类或者接口,或者两者都有,程序稳定性更好
  2. 变量的生命类型尽量是抽象类或接口,这样我们的变量引用和实际对象间就存在一个缓冲层,利于程序的扩展和优化
  3. 继承时遵循里氏替换原则

基本的依赖关系

接口 Appliance

/**
 * 所有的电器都必须要通电
 */
public interface Appliance {
    void electric();
}

Television 电视机

public class Television implements Appliance{
    @Override
    public void electric() {
        System.out.println("电视机通电了");
    }
}

Refrigerator 电冰箱

public class Refrigerator implements Appliance{
    @Override
    public void electric() {
        System.out.println("电冰箱通电了");
    }
}

Main

/**
 * 依赖倒转
 */
public class Main {
    public static void main(String[] args) {
        Appliance tel = new Television();
        Refrigerator ref = new Refrigerator();

        on(tel); // 给电视机通电
        on(ref); // 给电冰箱通电
    }

    /**
     * 接通电源
     * @param appliance 需要接通电源的电器
     */
    private static void on(Appliance appliance) {
        appliance.electric();
    }
}

依赖传递一:接口传递

相当于当今的智能化产品,将产品绑定到小爱同学、天猫精灵等终端,然后通过语音控制

接口 Appliance

/**
 * 所有的电器都必须要通电
 * 所有电器都必须要有开机和关机的功能
 */
public interface Appliance {
    void electric();
    void open();
    void close();
}

Television 电视机

public class Television implements Appliance{
    @Override
    public void electric() {
        System.out.println("电视机通电了");
    }

    @Override
    public void open() {
        System.out.println("电视机开机了");
    }

    @Override
    public void close() {
        System.out.println("电视机关机了");
    }
}

Refrigerator 电冰箱

public class Refrigerator implements Appliance{
    @Override
    public void electric() {
        System.out.println("电冰箱通电了");
    }

    @Override
    public void open() {
        System.out.println("电冰箱开机了");
    }

    @Override
    public void close() {
        System.out.println("电冰箱关机了");
    }
}

OpenAndClose 抽象出开关功能


public interface OpenAndClose {
    void open(Appliance appliance);
    void close(Appliance appliance);
}

XiaoMi 通过小爱同学开关电器

/**
 * 通过小爱同学来开关电器
 */
public class XiaoAi implements OpenAndClose{
    @Override
    public void open(Appliance appliance) {
        System.out.print("小爱同学: ");
        appliance.open();
    }

    @Override
    public void close(Appliance appliance) {
        System.out.print("小爱同学: ");
        appliance.close();

    }
}

TianMao 通过天猫精灵来开关电器

/**
 * 通过小米来开关电器
 */
public class TianMao implements OpenAndClose{

    @Override
    public void open(Appliance appliance) {
        System.out.print("天猫精灵: ");
        appliance.open();
    }

    @Override
    public void close(Appliance appliance) {
        System.out.print("天猫精灵: ");
        appliance.close();
    }
}

Main

/**
 * 依赖倒转
 */
public class Main {
    public static void main(String[] args) {
        Appliance tel = new Television();
        Refrigerator ref = new Refrigerator();
        on(tel);
        on(ref);

        tel.open(); // 打开电视机
        tel.open(); // 关闭电视机

        TianMao tianmao = new TianMao();
        XiaoAi xiaoai = new XiaoAi();
        
        // 天猫精灵打开电视机和电冰箱
        tianmao.open(tel);
        tianmao.open(ref);

        // 小爱同学关闭电视机和电冰箱
        xiaoai.close(tel);
        xiaoai.close(ref);
    }

    /**
     * 接通电源
     * @param appliance
     */
    private static void on(Appliance appliance) {
        appliance.electric();
    }
}

console:

电视机通电了
电冰箱通电了
电视机开机了
电视机开机了
天猫精灵: 电视机开机了
天猫精灵: 电冰箱开机了
小爱同学: 电视机关机了
小爱同学: 电冰箱关机了

依赖传递二:构造器传递

相当于产品配置的遥控器,每个遥控器只能单独控制那一个产品

RemoteControl

public class RemoteControl implements OpenAndClose2{
    private Appliance appliance;

    public RemoteControl(Appliance appliance) {
        this.appliance = appliance;
    }

    @Override
    public void open() {
        System.out.print("遥控器: ");
        this.appliance.open();
    }

    @Override
    public void close() {
        System.out.print("遥控器: ");
        this.appliance.close();
    }

    private void setAppliance(Appliance appliance) {
        this.appliance = appliance;
    }
}

Main

/**
 * 依赖倒转
 */
public class Main {
    public static void main(String[] args) {
        Appliance tel = new Television();
        Refrigerator ref = new Refrigerator();
        on(tel);
        on(ref);

        tel.open(); // 打开电视机
        tel.open(); // 关闭电视机

        TianMao tianmao = new TianMao();
        XiaoAi xiaoai = new XiaoAi();

        // 天猫精灵打开电视机和电冰箱
        tianmao.open(tel);
        tianmao.open(ref);

        // 小爱同学关闭电视机和电冰箱
        xiaoai.close(tel);
        xiaoai.close(ref);

        RemoteControl telControl = new RemoteControl(tel);
        RemoteControl refControl = new RemoteControl(ref);
        // 遥控器打开电视机和电冰箱
        telControl.open();
        refControl.open();
        // 遥控器关闭电视机和电冰箱
        telControl.close();
        refControl.close();
    }

    /**
     * 接通电源
     * @param appliance
     */
    private static void on(Appliance appliance) {
        appliance.electric();
    }
}

console

电视机通电了
电冰箱通电了
电视机开机了
电视机开机了
天猫精灵: 电视机开机了
天猫精灵: 电冰箱开机了
小爱同学: 电视机关机了
小爱同学: 电冰箱关机了
遥控器: 电视机开机了
遥控器: 电冰箱开机了
遥控器: 电视机关机了
遥控器: 电冰箱关机了

依赖传递三:set 传递

相当于一个万能遥控器,可以通过重新设置被遥控对象,达到遥控的目的

RemoteControl

package org.example.inversion;

public class RemoteControl implements OpenAndClose2{
    private Appliance appliance;

    public RemoteControl(Appliance appliance) {
        this.appliance = appliance;
    }

    @Override
    public void open() {
        System.out.print("遥控器: ");
        this.appliance.open();
    }

    @Override
    public void close() {
        System.out.print("遥控器: ");
        this.appliance.close();
    }

    public void setAppliance(Appliance appliance) {
        this.appliance = appliance;
    }
}

Main

public class Main {
    public static void main(String[] args) {
        Appliance tel = new Television();
        Refrigerator ref = new Refrigerator();
        on(tel);
        on(ref);

        tel.open(); // 打开电视机
        tel.open(); // 关闭电视机

        TianMao tianmao = new TianMao();
        XiaoAi xiaoai = new XiaoAi();

        // 天猫精灵打开电视机和电冰箱
        tianmao.open(tel);
        tianmao.open(ref);

        // 小爱同学关闭电视机和电冰箱
        xiaoai.close(tel);
        xiaoai.close(ref);

        RemoteControl telControl = new RemoteControl(tel);
        RemoteControl refControl = new RemoteControl(ref);
        // 遥控器打开电视机和电冰箱
        telControl.open();
        refControl.open();
        // 遥控器关闭电视机和电冰箱
        telControl.close();
        refControl.close();

        // 创建了一个万能遥控器
        RemoteControl whateverControl = new RemoteControl(null);
        System.out.println("设置遥控器为电视机遥控器");
        whateverControl.setAppliance(tel);
        whateverControl.open();
        whateverControl.close();
        System.out.println("设置遥控器为电冰箱遥控器");
        whateverControl.setAppliance(ref);
        whateverControl.open();
        whateverControl.close();
    }

    /**
     * 接通电源
     * @param appliance
     */
    private static void on(Appliance appliance) {
        appliance.electric();
    }
}

console

电视机通电了
电冰箱通电了
电视机开机了
电视机开机了
天猫精灵: 电视机开机了
天猫精灵: 电冰箱开机了
小爱同学: 电视机关机了
小爱同学: 电冰箱关机了
遥控器: 电视机开机了
遥控器: 电冰箱开机了
遥控器: 电视机关机了
遥控器: 电冰箱关机了
设置遥控器为电视机遥控器
遥控器: 电视机开机了
遥控器: 电视机关机了
设置遥控器为电冰箱遥控器
遥控器: 电冰箱开机了
遥控器: 电冰箱关机了

©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容