[Java基础]公共技术点之依赖注入

公共技术点之依赖注入

本文转载自:

http://a.codekk.com/detail/Android/%E6%89%94%E7%89%A9%E7%BA%BF/%E5%85%AC%E5%85%B1%E6%8A%80%E6%9C%AF%E7%82%B9%E4%B9%8B%E4%BE%9D%E8%B5%96%E6%B3%A8%E5%85%A5

1.依赖

如果在ClassA中,有ClassB的实例,则称ClassA对ClassB有一个依赖。例如下面类Human中用到一个Father对象,我们就说Human对类Father有一个依赖。

public class Human
{
    ...
    Father father;
    ...
    public Human()
    {
        father = new Father();
    }
}

仔细看这段代码我们会发现存在一些问题:
(1)如果要改变father生成方式,如需要new Father(String name)初始化father,需要修改Human代码;
(2)如何想测试不同Father对象对Human的影响很困难,因为father的初始化被写死在了Human的构造函数中;
(3)如果new Father()过程非常缓慢,单测时我们希望已经初始化好的father对象Mock掉这个过程也很困难

2.依赖注入

上面将依赖在构造函数中直接初始化是一种Hard init方式,弊端在于两个类不够独立,不方便测试。我们还有另外一种init方式,如下:

public class Human
{
    ...
    Father father;
    ...
    public Human(Father father)
    {
        this.father = father;
    }
}

上面代码中,我们将father对象作为构造函数的一个参数传入。在调用Human的构造方法之前外部就已经初始化好了Father对象。像这种非自己主动初始化依赖,而通过外部来传入依赖的方式,我们就称为依赖注入。

现在我们发现上面1中存在的两个问题都很好的解决了,简单的说依赖注入主要有两个好处:
(1)解耦,将依赖之间解耦
(2)因为已经解耦,所以方便做单元测试,尤其是Mock测试

3.Java中的依赖注入

依赖注入的实现有多种途径,而在Java中,使用注解是最常用的。通过在字段的声明前添加@Inject注解进行标记,来实现依赖对象的自动注入。

public class Human
{
    ...
    @Inject Father father;
    ...
    public Human()
    {
    }
}

上面这段代码看起来很神奇;只是增加了一个注解,Father对象就能自动注入了?这个注入过程是怎么完成的?

实质上,如果你只是写了一个@Inject注解,Father并不会被自动注入。你还需要使用一个依赖注入框架,并进行简单的配置。现在Java语言中较流行的依赖注入框架有Google Guice、Spring等,而在Android上比较流行的有RoboGuice、Dagger等。其中Dagger是我现在正在项目中使用的。如果有兴趣,你可以了解更多依赖注入和Dagger实现原理相关信息。

你可能感兴趣的:(java)