什么是依赖倒置原则

1、什么是依赖倒置原则

依赖倒置原则(Dependency Inversion Principle,DIP)是指高层模块不应该依赖于低层模块,它们都应该依赖于抽象。换句话说,具体类之间的依赖关系应该尽可能减少,而抽象类或接口之间的依赖关系应该尽可能增加。

这个原则的核心思想是将依赖关系从具体的实现类转移到抽象类或接口上。通过这种方式,高层模块和低层模块都可以依赖于抽象,而不是依赖于具体的实现类。这样做有几个好处:

  1. 降低耦合性:具体类之间的依赖关系减少,可以降低代码之间的耦合性,使得代码更加灵活和可维护。
  2. 可扩展性:通过将具体的实现类与使用类的耦合度降低到最小,我们可以方便地更换不同的实现类,而不需要修改使用类的代码。
  3. 灵活性:通过将具体的实现类注入到使用类中,我们可以灵活地选择不同的实现方案。例如,在不同的场景下使用不同的数据库连接或者数据访问方式。

简单来说,依赖倒置原则就是让我们不要直接依赖于具体的类,而是依赖于抽象的接口或抽象类。这样可以让我们的代码更加灵活、可维护和可扩展。

2、我们跟着一个例子更好的理解一下

假设我们有一个应用程序,它有一个“用户界面”组件和一个“数据存储”组件。用户界面组件负责处理用户输入和展示数据,而数据存储组件则负责数据的存储和检索。

// 定义一个抽象的接口  
public interface DataAccess {  
    void saveData(String data);  
    String readData();  
}  
  
// 实现抽象接口的具体类  
public class MySQLDataAccess implements DataAccess {  
    @Override  
    public void saveData(String data) {  
        // 具体实现细节  
    }  
  
    @Override  
    public String readData() {  
        // 具体实现细节  
        return null;  
    }  
}  
  
public class MongoDBDataAccess implements DataAccess {  
    @Override  
    public void saveData(String data) {  
        // 具体实现细节  
    }  
  
    @Override  
    public String readData() {  
        // 具体实现细节  
        return null;  
    }  
}  
  
// 定义一个类,依赖于抽象接口  
public class UserInterface {  
    private DataAccess dataAccess;  
  
    // 通过构造函数注入依赖项  
    public UserInterface(DataAccess dataAccess) {  
        this.dataAccess = dataAccess;  
    }  
  
    public void saveData(String data) {  
        dataAccess.saveData(data);  
    }  
  
    public String readData() {  
        return dataAccess.readData();  
    }  
}

在这个例子中,我们定义了一个抽象的DataAccess接口,然后有两个具体类MySQLDataAccessMongoDBDataAccess实现了这个接口。接着,我们定义了一个UserInterface类,它依赖于抽象的DataAccess接口,而不是具体的实现类。最后,我们在构造函数中注入了依赖项,这样就可以让UserInterface类依赖于抽象的接口,而不是具体的实现类。

3、怎么将一个具体的DataAccess实现类(例如MySQLDataAccess)注入到UserInterface类中

在上面的代码例子中,我们将一个具体的DataAccess实现类(即MySQLDataAccess)注入到UserInterface类中的方式是通过构造函数注入。我们为UserInterface类的构造函数添加一个DataAccess类型的参数,并在构造方法中接收这个参数。

下面是将MySQLDataAccess注入到UserInterface类中的代码示例:

java复制代码
	// 创建MySQLDataAccess实例  

	MySQLDataAccess mysqlDataAccess = new MySQLDataAccess();  

	  

	// 创建UserInterface实例,并将MySQLDataAccess实例注入到构造函数中  

	UserInterface userInterface = new UserInterface(mysqlDataAccess);

在这个例子中,我们首先创建了一个MySQLDataAccess实例,并将其作为参数传递给UserInterface类的构造方法。这样,当UserInterface类的实例被创建时,就会接收一个已经实例化的MySQLDataAccess对象,从而实现了依赖注入。

需要注意的是,在实际的应用程序中,我们通常会使用依赖注入框架(如Spring)来管理组件之间的依赖关系。这样可以避免手动创建和注入依赖项,提高代码的可维护性和可扩展性。

4、结论

在没有采用依赖倒置原则的情况下,用户界面组件可能会直接依赖于数据存储组件。这意味着用户界面组件需要知道数据存储组件的具体实现细节,例如数据库的URL、用户名和密码等。这会导致用户界面组件和数据存储组件紧密耦合,不利于代码的可维护性和可复用性。

采用依赖倒置原则后,我们可以创建一个抽象的“数据访问接口”,然后让数据存储组件实现这个接口。同时,用户界面组件只需要依赖于这个抽象的数据访问接口,而不需要知道具体的实现细节。这样,即使我们更换了数据存储组件(例如从MySQL数据库更换为MongoDB数据库),用户界面组件也不需要进行任何修改,因为它们只依赖于抽象的数据访问接口。

这个例子中,依赖倒置原则的实践主要体现在以下几点:

  1. 定义抽象的数据访问接口,而不是针对具体的数据存储组件。
  2. 让数据存储组件实现这个接口,而不是让用户界面组件依赖于具体的数据存储组件。
  3. 使用外部容器(例如Spring)来管理组件之间的依赖关系,而不是让组件自己去创建和管理其他组件。

通过这种方式,我们可以降低代码之间的耦合,提高代码的可维护性、可读性和可复用性。

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