首页 > 编程笔记 > Java笔记

使用依赖倒置原则解决实际问题

下面来看一个实例,以 新宝库的教程为例,首先创建一个学员类 ZhangSan。
package net.biancheng.c.dip;

public class ZhangSan {
    public void studyJava() {
        System.out.println("张三正在学习Java教程");
    }

    public void studyCplus() {
        System.out.println("张三正在学习C++教程");
    }
}
然后编写客户端测试代码并调用。
public static void main(String[] args) {
    ZhangSan zhangSan = new ZhangSan();
    zhangSan.studyJava();
    zhangSan.studyCplus();
}
目前张三正在学习 Java 和 C++ 教程,随着学习兴趣的暴涨,张三还想学习 Python 教程。这时就需要业务扩展,代码需要从底层开始修改,即:在 ZhangSan 类中添加 studyPython() 的方法,在高层(这里指的是 main() 方法)也要追加调用该方法。这样发布系统,实际上是非常不稳定的,在修改代码的同时会带来意想不到的风险。

下面我们优化代码,首先创建教程的抽象接口 Course。
package net.biancheng.c.dip;

public interface Course {
    void study();
}
然后编写 JavaCourse 类。
package net.biancheng.c.dip;

public class JavaCourse implements Course {

    @Override
    public void study() {
        System.out.println("张三在学习Java教程");
    }
}
同样,编写 CplusCourse 类。
package net.biancheng.c.dip;

public class CplusCourse implements Course {

    @Override
    public void study() {
        System.out.println("张三在学习C++教程");
    }
}
最后修改 ZhangSan 类。
public class ZhangSan {
    public void study(Course course) {
        course.study();
    }
}
客户端测试代码如下。
public static void main(String[] args) {
    ZhangSan zhangSan = new ZhangSan();
    zhangSan.study(new JavaCourse());
    zhangSan.study(new CplusCourse());
}
这个时候再看代码,无论张三要学习什么新教程,只需要建一个类,通过传参的方式告诉 ZhangSan 即可,而不需要修改底层代码。有的小伙伴可能已经发现了,这是大家非常熟悉的依赖注入。

注入的方式还有构造方法注入和 Setter 注入。下面来看构造方法注入。
public class ZhangSan {
    private Course course;

    public ZhangSan(Course course) {
        this.course = course;
    }

    public void study(Course course) {
        course.study();
    }
}
客户端代码如下,将 JavaCourse 对象作为 ZhangSan 对象的构造参数注入。
public static void main(String[] args) {
    ZhangSan zhangSan = new ZhangSan(new JavaCourse());
    zhangSan.study();
}
根据构造方法注入,调用时每次都要创建实例。如果 ZhangSan 是全局单例,只能选择 Setter 注入方式,继续修改 ZhangSan 类的代码。
package net.biancheng.c.dip;

public class ZhangSan {
    private Course course;

    public void setCourse(Course course) {
        this.course = course;
    }

    public void study() {
        course.study();
    }
}
来看客户端代码,调用 ZhangSan 对象的 setCourse() 方法,将 JavaCourse 对象作为参数。
public static void main(String[] args) {
    ZhangSan zhangSan = new ZhangSan();
    zhangSan.setCourse(new JavaCourse());
    zhangSan.study();

    zhangSan.setCourse(new CplusCourse());
    zhangSan.study();
}
最终得到的类图如下。
类图

所有教程

优秀文章