控制反转(IoC)和依赖注入(DI)

前言

学习 Spring 的第一个任务就是需要理解控制反转 (IoC) 和依赖注入 (DI) 是什么。

接下来围绕这两个概念讨论几个问题:

什么是控制反转

什么是依赖注入

Spring 中的依赖注入是如何应用的

控制反转

控制反转 (Ioc) 并不是一种技术,而是一种思想。

用一个例子来说明什么是控制反转:

某天,老刘打电话给老李询问孩子就业的的问题,老李说,你别打来了,等事情办好了我自然会让人事部打电话通知你。

此时,这就是一个控制反转。打电话的控制权在自己手上,经过反转控制,控制权交给了人事部。

这样不太明白请接下来继续看。

进一步说:

当一个对象 A 依赖于另外一个对象 B,那么对象 A 就需要创建一个对象 B 才能完成任务,这是主动创建。

控制反转就是将主动创建这个任务交给别人,这样,对象 A 就不需要自己创建对象 B 就能完成任务。

也就是说,对象 A 在运行到某个时候需要创建对象 B 才能继续工作,此时,只能由对象 A 来创建对象 B,这样,控制权在 A 的手上。

经过 IoC 容器控制反转之后,情况就变化了:因为有 IoC 容器加入,A 对象运行到需要创建对象 B 的时候,只需要将创建对象的任务交给容器,IoC 容器会创建一个对象 B 注入到 A 所需要的地方。

这样,控制权互换了一下,原来是创建对象 B 的主动行为现在变成了由别人注入成为了被动行为。这就是控制反转

海尔公司作为一个电器制商需要把自己的商品分销到全国各地,但是发现,不同的分销渠道有不同的玩法,于是派出了各种销售代表玩不同的玩法,随着渠道越来越多,发现,每增加一个渠道就要新增一批人和一个新的流程,严重耦合并依赖各渠道商的玩法。实在受不了了,于是制定业务标准,开发分销信息化系统,只有符合这个标准的渠道商才能成为海尔的分销商。让各个渠道商反过来依赖自己标准。反转了控制,倒置了依赖。

依赖注入

先来看看一段源码


public class A {
    ...
    B b;
    ...
    public A() {
        b = new B();
    }
}

这个代码似乎没有任何问题,事实上运行之后也确实没有任何问题。

我们分析一下这段代码

首先 Class A 中有一个 B 的实例,也就是说, Class A 依赖于 Class B

我们可以说: Class A 对 Class B 有一个依赖。

在 Class A 的构造方法中,创建了一个对象 B

如此以来,我们是不是发现,Class A 主动创建了对象 B

这样带来了问题:

  • 如果某一天改变了对象 B 的初始化方式,我们需要在每个创建对象 B 的地方做修改。
  • 耦合高

再来看看下面的代码:


public class A {
    ...
    B b;
    ...
    public A(B b) {
        this.b = b;
    }
}

这段代码与上面的代码有了新一步的飞跃!

我们看见,创建对象的方式并不是由 A 主动创建,而是由别人为他创建,究竟是谁创建 A 不需要管。

这就是一个依赖注入,依赖注入的工作交给了 IoC 容器。

Spring 的依赖注入


public class A {
    ...
    B b;
    ...
    public void setB(B b) {
    this.b = b;
    }
}

class A {
    ...
    String c;
    ...
    public void setB(String c) {
    this.c = c;
    }
}

现在我们将创建对象 B 的任务交给了 Spring

所以在 Spring 进行配置:


    
        
            
        
    
    
        
            test
        
    


这样,Spring 就将对象 B 创建好并注入在对象 A 需要使用的地方了。

总结

控制反转是一种思想,将工作交给其他人,自己不需要管理,只依赖接口,不依赖接口的具体实现。

依赖注入是一种设计模式,是实现控制反转的一种方式,用来解耦。

Spring 是一个 IoC 容器。

你可能感兴趣的:(控制反转(IoC)和依赖注入(DI))