Jetpack Hilt 依赖注入框架
什么是依赖注入
一个类里面有一个变量,这个变量就是这个类的依赖。然后通过外部注入对这个变量进行赋值,这种就叫做依赖注入。
Hilt 是什么
Hilt 是 Android 的依赖注入库,基于 Dagger 构建。Hilt 专门为 Android 打造,创建了一组标准的组件和作用域。这些组件会自动集成到 Android 程序中的生命周期中。在使用的时候可以指定使用的范围,使其作用在对应的生命周期当中。
Hilt 常用的注解含义
@HiltAndroidApp
@HiltAndroidApp 将会触发 Hilt 的代码生成,作为程序依赖项的基类。
生成的 Hilt 依附于 Application 的生命周期,它是 App 的父组件,提供访问其他组件的依赖。
在 Application 中配置好后,就可以使用 Hilt 提供的组件了;组件包含 Application、Activity、Fragment、View、Service 等。
创建一个依赖容器,该容器遵循 Android 的生命周期,目前支持的类型是:Activity, Fragment, View, Service, BroadcastReceiver。
@Inject
使用 @Inject 来告诉 Hilt 如何提供该类的实例,常用于构造方法、非私有字段、方法中。
Hilt 有关如何提供不同类型的实例信息也称之为绑定。
@Module
Module 是用来提供一些无法用 @Inject 构造的依赖,如第三方库、接口、Build 模式的构造等。
使用 @Module 注解的类,需要使用 @InstallIn 注解指定 module 的范围。
增加了 @Module 注解的类,其实代表的就是一个模块,并通过指定的组件来告诉在那个容器中可以使用绑定安装。
@InstallIn
使用 @Module 注入的类,需要使用 @InstallIn 注解指定 module 的范围。
例如使用 @InstallIn(ActivityComponent::class) 注解的 module 会绑定到 Activity 的生命周期上。
@Provides
常用于被 @Module 注解标记类的内部方法上,并提供依赖项对象。
@EntryPoint
Hilt 支持最常见的 Android 类 Application、Activity、Fragment、View、Service、BroadcastReceiver 等等,但是您可能需要在 Hilt 不支持的类中执行依赖注入,在这种情况下可以使用 @EntryPoint 注解进行创建,Hilt 会提供相应的依赖。
Hilt 中的组件 (Component)
使用 @Module 注解的类,需要使用 @InstallIn 注解来指定 module 的范围。
例如 @InstallIn(ApplicationComponent::class) 注解的 Module 就会绑定到 Application 的生命周期上。
Hilt 提供了以下组件来绑定依赖与对应 Android 类的活动范围:
| Hilt 组件 | 对应 Android 类活动的范围 |
|---|
| ApplicationComponent | Application |
| ActivityRetainedComponent | ViewModel |
| ActivityComponent | Activity |
| FragmentComponent | Fragment |
| ViewComponent | View |
| ViewWithFragmentComponent | View annotated with @WithFragmentBindings |
| ServiceComponent | Service |
Hilt 没有为 broadcast receivers 提供组件,因为 Hilt 直接从 ApplicationComponent 中注入 broadcast receivers。
Hilt 中组件的生命周期
Hilt 会根据相应的 Android 类生命周期自动创建和销毁组件的实例,对应关系如下:
| Hilt 提供的组件 | 创建对应的生命周期 | 结束对应的生命周期 | 作用范围 |
|---|
| ApplicationComponent | Application#onCreate() | Application#onDestroy() | @Singleton |
| ActivityRetainedComponent | Activity#onCreate() | Activity#onDestroy() | @ActivityRetainedScope |
| ActivityComponent | Activity#onCreate() | Activity#onDestroy() | @ActivityScoped |
| FragmentComponent | Fragment#onAttach() | Fragment#onDestroy() | @FragmentScoped |
| ViewComponent | View#super() | View destroyed | @ViewScoped |
| ViewWithFragmentComponent | View#super() | View destroyed | @ViewScoped |
| ServiceComponent | Service#onCreate() | Service#onDestroy() | @ServiceScoped |
如何使用 Hilt
1. 添加依赖
// build.gradle (Project level)
buildscript {
dependencies {
classpath 'com.google.dagger:hilt-android-gradle-plugin:2.45'
}
}
// build.gradle (App level)
plugins {
id 'com.android.application'
id 'kotlin-kapt'
id 'dagger.hilt.android.plugin'
}
dependencies {
implementation 'com.google.dagger:hilt-android:2.45'
kapt 'com.google.dagger:hilt-android-compiler:2.45'
}
2. 配置 Application
class BaseApplication : Application() {
override fun onCreate() {
super.onCreate()
}
}
在 AndroidManifest.xml 中将 BaseApplication 设置为 application 标签的 name 属性,并添加 @HiltAndroidApp 注解。
使用 Hilt 进行依赖注入
class HiltTest {
fun hiltTest() {
Log.e("----------->", "hiltTest")
}
}
class BaseApplication : Application() {
@Inject
lateinit var hiltTest: HiltTest
override fun onCreate() {
super.onCreate()
}
}
Hilt 在 Android 组件中的使用
- 如果使用 @AndroidEntryPoint 注解 Android 类,还必须注解依赖他的 Android 类;
- 例如:给 Fragment 使用 @AndroidEntryPoint 后,则还需要给 Fragment 依赖的 Activity 添加 @AndroidEntryPoint,否则会出现异常;
- @AndroidEntryPoint 不能写在抽象类上;
- @AndroidEntryPoint 注解仅仅支持 ComponentActivity 的子类,例如 Fragment、AppCompatActivity 等等。
@AndroidEntryPoint
class HomeNavigationActivity : AppCompatActivity() {
override fun onCreate(savedInstanceState: Bundle?) {
super.onCreate(savedInstanceState)
setContentView(R.layout.activity_home_navigation)
}
}
@AndroidEntryPoint
class FragmentOne : Fragment() {
@Inject
lateinit var hiltTest: HiltTest
override fun onViewCreated(view: View, savedInstanceState: Bundle?) {
super.onViewCreated(view, savedInstanceState)
hiltTest.hiltTest()
}
}
Hilt 和第三方组件的使用
如果需要在项目中注入第三方依赖,可以使用 @Module 注解。使用 @Module 在注解的普通类,在其中创建第三方依赖的对象即可。
使用了 @Module 的类,相当于是一个模块,常用于创建依赖对象(如 OkHttp、Retrofit 等)。
使用 @Module 的类,需要使用 @InstallIn 指定此 module 的范围,会绑定到对应 Android 类的生命周期上。
@Provides,常用于被 @Module 注解标记类的内部方法,并提供依赖项对象。
@Module
@InstallIn(ApplicationComponent::class)
object TestModule {
@Provides
fun bindHiltTest(): HiltTest {
return HiltTest()
}
@Provides
@Singleton
fun bindSingTest(): Test {
return Test()
}
}
使用如下:
class SomeClass {
@Inject
lateinit var hiltTest: HiltTest
@Inject
lateinit var test1: Test
@Inject
lateinit var test2: Test
}
其中 bindSingTest 只会被调用一次,@Singleton 相当于是一个单例。
Hilt 和 ViewModel 的使用
使用之前需要在 app.build 下添加对 ViewModel 的支持。
implementation 'androidx.hilt:hilt-lifecycle-viewmodel:1.0.0-alpha03'
kapt 'androidx.hilt:hilt-compiler:1.0.0-alpha03'
- 通过 @ViewModelInject 注解进行构造注入。
- SavedStateHandle 使用 @Assisted 注解。
class HomeContentViewModel(
private val response: HomeContentRepository,
@Assisted val state: SavedStateHandle
) : ViewModel() {
private val liveData by lazy { MutableLiveData<String>() }
val testLiveData: LiveData<String> by lazy { liveData }
fun requestBaiDu() {
liveData.postValue(response.requestBaidu())
}
}
- 通过 @Inject 进行注入,在 ViewModel 中不需要手动创建其对象。
class HomeContentRepository @Inject constructor() {
suspend fun requestBaidu(): String {
return "Response from Baidu"
}
}
class HomeContentActivity : AppCompatActivity(){
private val viewModel: HomeContentViewModel by viewModels<HomeContentViewModel>()
override fun onCreate(savedInstanceState: Bundle?) {
super.onCreate(savedInstanceState)
setContentView(R.layout.activity_home_content)
viewModel.requestBaiDu()
viewModel.testLiveData.observe(this) {
Toast.makeText(this, it, Toast.LENGTH_SHORT).show()
}
}
}
Hilt 和 Room 的使用
这里需要用到 @Module 注解,使用 @Module 注解的普通类,在其中提供 Room 的实例。并且使用 @InstallIn 来声明作用范围。
@InstallIn(ApplicationComponent::class)
object RoomModule {
@Provides
@Singleton
fun provideAppDataBase(application: Application): AppDataBase {
return Room.databaseBuilder(
application,
AppDataBase::class.java,
"knif.db"
).fallbackToDestructiveMigration().build()
}
@Provides
@Singleton
fun providerUserDao(appDataBase: AppDataBase): UserDao {
return appDataBase.getUserDao()
}
}
我们给 providerUserDao 使用了 @Provides 注解和 @Singleton 注解,是为了告诉 Hilt,当使用 UserDao 时需要执行 appDataBase.getUserDao()。
而在调用 appDataBase.getUserDao() 时需要传入 AppDataBase,这时就会调用上面的方法 provideAppDataBase 了,因为这个方法也是用了 @Provides 注解。
并且这两个方法都是单例,只会调用一次。
使用如下:
class SomeFragment : Fragment() {
@Inject
lateinit var userDao: UserDao
}
到现在为止,就可以在任意地方获取到 UserDao,并且不用手动创建实例。
使用 @Binds 进行接口注入
Binds:必须注释一个抽象函数,抽象函数的返回值是实现的接口。通过添加具有接口实现类型的唯一参数来指定实现。
首先需要一个接口,和一个实现类。
interface User {
fun getName(): String
}
class UserImpl : User {
override fun getName(): String {
return "345"
}
}
接着就需要新建一个 Module,用来实现接口的注入。
@InstallIn(ApplicationComponent::class)
abstract class UserModule {
@Binds
abstract fun getUser(userImpl: UserImpl): User
}
注意:这个 Module 是抽象的。
使用如下:
class FragmentOne : Fragment() {
@Inject
lateinit var user: User
}
使用 @Qualifier 提供同一接口,不同的实现
还是上面的 User 接口,有两个不同的实现,如下:
class UserAImpl : User {
override fun getName(): String {
return "345"
}
}
class UserBImpl : User {
override fun getName(): String {
return "Lv"
}
}
接着定义两个注解。
@Qualifier
annotation class A
@Qualifier
annotation class B
然后修改 Module,在 module 中用来标记相应的依赖。
@InstallIn(ApplicationComponent::class)
abstract class UserAModule {
@A
@Singleton
@Binds
abstract fun getUserA(userImpl: UserAImpl): User
}
@Module
@InstallIn(ActivityComponent::class)
abstract class UserBModule {
@B
@ActivityScoped
@Binds
abstract fun getUserB(userImpl: UserBImpl): User
}
这里用了两个不同的 module,并且对应两个不同的 component,一个是 Application,另一个是 Activity。
最后使用如下:
class FragmentOne : Fragment() {
@A
@Inject
lateinit var userA: User
@B
@Inject
lateinit var userB: User
}
遇到的问题
在使用 @AndroidEntryPoint 注解的时候,需要在 Fragment 和 Activity 都使用这个注解。
但是如果 Activity 和 Fragment 没在同一个 Module 中,就会报错。
对于组件化的项目来说,这种情况就比较难受。
主要问题之一是,通过在 Hilt 中发现模块的方式,无法区分哪些模块属于应用中的组件(如果他们确实使用过 Hilt)以及库或其他库中的组件。
另一个问题是,它将预先构建的组件层次结构变得复杂和混乱。就将你的库中所有活动一样,使父级成为 ApplicationComponent 也没有意义,因为你没有将组件放入 Application。
同样,如果一个仅包含片段库并托管在应用程序的活动中,那可能会遇到类似的情况,你希望库片段是独立的,但让 FragmentComponent 对象作为 ActivityComponent 并没有意义。
Hilt 好处
- 降低 Android 开发者使用依赖注入框架的上手成本。
- 内部有一套标准的组件和作用域,对范围进行声明后,只能使用在指定的作用域中使用这个类,并且提供声明周期的管理,会自动释放不在使用的对象,减少资源的过度使用,提供代码的可重用性。
- 使用起来简单,告别繁琐的 new... 这种流程,只需要添加注解即可。提高了代码的可读性,构建简单,耦合变低,容易测试。
- 最大的好处就是管理他们的生命周期,只能在对应的范围内进行使用。感觉非常好。