概述
Dagger 的名字取自有向无环图 DAG (directed acyclic graph),因为程序里的依赖关系拼接起来就是一个或者多个有向无环图。
首先理解一下什么是依赖注入。一个类 UserRepository 中有一个 UserRemoteDataSource 类型的属性, 那 UserRemoteDataSource 便是 UserRepository 的依赖,初始化这个依赖可以有两种方法,一种是在类内部自己初始化,另一种是由外部初始化(即是依赖注入,关键在于初始化是谁做的)。
这种由外部初始化的方式都可以叫做依赖注入,而 Dagger 则为依赖注入提供了一种更简单的方式。
基础用法
接下来用 Dagger2 实现如下图所示的依赖关系:
首先添加 Dagger2 的依赖:
1 | apply plugin: 'kotlin-kapt' |
然后定义 UserRepository 类:
1 | class UserRepository constructor( |
这里通过在构造器上添加 @Inject
注解告知 Dagger 如何创建 UserRepository 实例,它的依赖项为 UserLocalDataSource 和 UserRemoteDataSource, 为了让 Dagger 知道如何创建这两个依赖项实例,也需使用 @Inject
注解:
1 | interface DataSource { |
这样 Dagger 便会知道如何创建它们。
然后在 ViewModel 中也可以通过这种方式依赖 UserRepository:
1 | class LoginViewModel constructor(private val userRepository: UserRepository) { |
接下来在 Activity 中怎么引用 LoginViewModel 实例呢?显然它不能跟上面一样在构造方法中使用 @Inject
注解了,因为 Activity 实例化是由系统完成的,无法由 Dagger 去进行实例化,因此这里不能由构造函数注入 LoginViewModel, 而应该使用属性注入(注意:注入的属性不能为私有属性):
1 | class LoginActivity : AppCompatActivity() { |
光是这样显然是不够的,我们还需要告知 Dagger 要求注入依赖项的对象是谁(LoginActivity),在此之前就得先看看 Dagger Component 的概念。
Component
Dagger 可以在项目中创建一个有向无环图(依赖关系),然后它可以从该图中了解在需要这些依赖项时从何处获取它们。为了让 Dagger 执行此操作,我们需要创建一个接口,并使用 @Component
为其添加注解,在上面的基础用法里,Dagger 需要知道 LoginActivity 中的 LoginViewModel 依赖应该怎么创建:
属性注入
对于属性注入,Dagger 需要让 LoginActivity 访问该图才能提供所需的 LoginViewModel 依赖实例,为此应在 Component 接口中提供一个函数,让该函数将请求注入的对象作为参数。
1 |
|
@Component
会让 Dagger 生成一个容器,其中应包含满足其提供的类型所需的所有依赖项,这称为 Dagger 组件,它包含一个图,其中包括 Dagger 知道如何提供的对象及其各自的依赖项。在 build 项目时 Dagger 会生成 ApplicationComponent 接口的实现:DaggerApplicationComponent。Dagger 通过其注解处理器(处理 @Inject
等)创建了一个依赖关系图,包含了这些依赖之间的关系,并且将 ApplicationComponent 接口中我们加入的方法(inject 方法,方法名可以随意)涉及的依赖作为入口点。
在声明了 inject 方法后,我们可以这样实现注入:
1 | class LoginActivity : AppCompatActivity() { |
构造器注入
在上面的 LoginActivity 中,如果不通过属性注入 LoginViewModel 依赖的话,还可以通过构造器注入的方式,这里需要在 @Component
接口内定义返回所需类(即 LoginViewModel)的实例的函数。
1 |
|
然后在 LoginActivity 中给 viewModel 赋值:
1 | class LoginActivity : AppCompatActivity() { |
Module
除了 @Inject
之外,还有一种方法可告知 Dagger 如何提供类实例 —— @Module
和 @Provides
。
假设将上例中的 UserRemoteDataSource 构造方法改一下,其依赖于 UserService 类:
1 | class UserRemoteDataSource constructor(private val userService: UserService) : DataSource { |
可以看到 UserService 对外的创建方法不是通过构造函数,而是通过其 Builder 类来实例化的,那这里就不能使用 @Inject
来进行注入依赖了,在实际项目中这样的场景也不少,我们可以使用 @Module
注解来定义 Dagger 模块,使用 @Provides
注解定义依赖项:
1 |
|
上面的 Module 注解用来告知 Dagger 这是一个 Module 类,而 Provides 注解用来告诉 Dagger 怎么创建该方法返回的类型的实例。这里如果要创建的 UserService 实例需要依赖其他对象,可以将依赖作为方法参数传入,当然该依赖也需要使用 Inject 等注解来告知 Dagger 怎么去创建它。
作用域限定
在上面通过 DaggerApplicationComponent.create().viewModel()
创建的 ViewModel 实例,同一个 DaggerApplicationComponent 对象调用多次 viewModel() 方法,Dagger 默认会创建多个不同的 ViewModel 实例。可以验证:
1 | val component = DaggerApplicationComponent.create() |
如果需要让 Dagger 中同一个 DaggerApplicationComponent 对象始终返回同一 viewModel 实例,可以使用作用域注解(@Singleton
)将某个对象的生命周期限定为其 Component 对象的生命周期,这意味着每次获取时都会使用该依赖项的同一实例。
1 |
|
@Singleton
是 javax.inject
软件包随附的唯一一个作用域注解,也可以创建并使用自定义作用域注解,使用方式跟 @Singleton
一样:
1 |
|
也可以对 Module 中提供的依赖使用与 Component 一样的作用域注解,这样 Provides 方法中返回的依赖也就与 Component 对象有了一样的生命周期了:
1 |
|
注意:使用作用域注解的依赖只能在带有相同作用域注解的 Component 中使用。
在 Android 中我们通常会创建一个位于 Application 类中的 Dagger 图,这样就将该图附加到应用的生命周期里了:
1 | class MainApplication : Application() { |
于是上面与该 ApplicationComponent 使用同一 @Singleton
注解的依赖项也将获得与其一致的生命周期,通过该 ApplicationComponent 对象中的方法获取到的依赖对象(applicationGraph.viewModel()等),其生命周期将与其一致。
作用域限定规则:
- 如果某个依赖类标记有作用域注解,该类型就只能由带有相同作用域注解的组件 Component 使用。
- 如果某个组件 Component 标记有作用域注解,该组件就只能提供带有该注解的类型或不带注解的类型。
- 子组件(后面会讲到)不能使用其某一父组件使用的作用域注解。
Subcomponent
为什么使用子组件
上面我们在 MainApplication 应用类中创建了一个 DaggerApplicationComponent 实例,当通过这个 applicationGraph 实例去获取依赖的时候,对于使用 Singleton 注解的依赖(注意 Component 也需要用 Singleton 注解),则其会跟应用生命周期一致(蕾丝于全局单例)。
如需将 LoginViewModel 的作用域限定为 LoginActivity 的生命周期,我们可以在 LoginActivity 中创建 DaggerApplicationComponent 实例属性,然后通过这个实例去获取 LoginViewModel 对象,由于此时这个 DaggerApplicationComponent 实例的生命周期跟 LoginActivity 绑定,则 LoginViewModel 的作用域也会被限定为 LoginActivity 的生命周期:
1 | class LoginActivity : AppCompatActivity() { |
通过上面这个方式,可以将 LoginViewModel 的作用域限定为 LoginActivity 的生命周期,但由于 DaggerApplicationComponent 被定义为应用全局的依赖图,而这里我们所需的依赖仅是 LoginActivity 相关的,且又要限制 LoginViewModel 的作用域,为此又在 LoginActivity 中反复创建 DaggerApplicationComponent 实例,显得不那么好看。而且 LoginViewModel 的依赖应该只有登陆这些场景才会使用,可以避免将其放到应用全局的依赖图里。因此这里可以考虑使用子组件(子依赖图),应用组件 DaggerApplicationComponent 中只放一些共有的依赖图,不同的应用场景考虑增加子图,子图存放特有应用/业务场景的依赖,如 LoginViewModle 就放在登陆子图 LoginComponent 中。
子组件是继承并扩展父组件的对象图的组件,因此,父组件中提供的所有对象也将在子组件中提供,这样子组件中的对象就可以依赖于父组件提供的对象(共有),父组件向子组件提供的对象的作用域仍限定为父组件的生命周期。
创建子组件
1、定义子组件
1 |
|
2、创建子组件的模块
1 |
|
3、将声明子组件的模块添加到 ApplicationComponent 中
1 |
|
注意,这里 ApplicationComponent 之前的 inject 方法删除了,因为 inject 放到了专门的登陆子组件中。增加了一个返回 UserRepository 实例的 repository() 方法,如果不加这个方法且 UserRepository 没有跟 ApplicationComponent 一样用 Singleton 注解标识的话,UserRepository 就会被包含到 LoginComponent 依赖图中,因为它被 LoginViewModel 引用了,这是目前使用它的唯一位置,而 LoginViewModel 是 LoginComponent 子组件中的依赖。现在的依赖关系如下图:
4、使用
1 | class LoginActivity : AppCompatActivity() { |
LoginComponent 是在 Activity 的 onCreate() 方法中创建的,将随着 Activity 的销毁而被隐式销毁。这样它就具备了跟 LoginActivity 一致的生命周期了。
作用域限定
可以创建自定义作用域注解,并使用该作用域为 LoginComponent 和 LoginViewModel 添加注解。此时如果存在两个需要 LoginViewModel 依赖的 Fragment 页面,则我们在这两个 Fragment 中注入的 LoginViewModel 都是同一个实例(注意不能使用 Singleton 注解,因为其已经被父组件占用了)。
可以将该注解命名为 ActivityScope, 这里不使用 LoginScope, 因为该注解也可以被其他蕾丝的场景(同级组件)使用。
1 |
|
然后在这两个 Fragment 中可以注入 LoginViewModel 实例:
1 | class LoginAFragment: Fragment() { |
组件依赖
一个 Component 可以通过设置 dependencies 依赖另一个 Component 组件,然后它便可获取到所依赖的 Component 中暴露的依赖。
我们首先创建一个零件组件:
1 |
|
这里使用 Singleton 注解将 Memory 和 Cpu 限制到 PartComponent 零件组件的作用域里。
接着假设一个 Computer 类需要依赖 Cpu 和 Memory:
1 | // 看实际场景,可加可不加 |
然后定义一个依赖 PartComponent 组件的 ComputerComponent 组件:
1 | // 需要加作用域,在新版本 Dagger 中没加作用域的话,依赖的 PartComponent 组件如果有作用域注解则会编译报错 |
在这里 ComputerComponent 依赖 PartComponent 组件,因此它可获取到 PartComponent 接口中暴露的依赖:
1 | class MainActivity : AppCompatActivity() { |
这里我们如果将 PartComponent 接口中暴露依赖的两个方法删掉,那么 ComputerComponent 便不能访问到 Cpu 和 Memory 了,编译报错,因为它会直接尝试去找 Computer 的两个依赖,然后发现这两个依赖被 Singleton 注解标识了,而 ComputerComponent 自身是 ActivityScope 作用域的,因此编译报错(之前我们说依赖如果有作用域注解的话,其作用域注解要和组件的一致)。
关于组件依赖还可举例,依旧用上面的 Cpu 和 Memory 类:
1 | class Cpu { |
这里我们通过 Module 的方式提供依赖。接着在 PartComponent 组件中声明模块,并暴露出获取依赖的方法。
1 |
|
然后跟上面一样让 ComputerComponent 去依赖 PartComponent 并注入 Computer,这里可以视情况增加作用域限定。
总结
Google 在 Jetpack 包中增加了 Hilt 组件,它是专门针对 Android 平台做的一个依赖注入库,底层依旧是基于 Dagger, 因此学习 Dagger 还是挺有必要的。Hilt 并不是对 Dagger 做了优化,而是针对 Android 开发制定了一套场景化的规则,刚学习了 Dagger 的一些用法,后续有时间接着学习一下 Hilt 及其实现原理。
Dagger 的用法会比较复杂一点,这篇文章结合了我自己的一些理解和用例,讲解了一下 Dagger 的基础用法。文中内容如有错误欢迎指出,共同进步!觉得不错的留个赞再走哈~
参考: https://developer.android.com/training/dependency-injection/dagger-android