上手指南 | Jetpack Hilt 依赖注入框架

简介: 上手指南 | Jetpack Hilt 依赖注入框架

依赖注入是什么


一个类里面有一个变量,这个变量就是这个类的依赖。然后通过外部注入对这个变量进行赋值,这种就叫做依赖注入。


Hilt 是什么


Hilt 是 Android 的依赖注入库,其实是基于 Dagger 。可以说 Hilt 是专门为 Andorid 打造的。


Hilt 创建了一组标准的 组件和作用域。这些组件会自动集成到 Android 程序中的生命周期中。在使用的时候可以指定使用的范围,事情作用在对应的生命周期当中。


Hilt 常用的注解的含义


@HiltAndroidApp


@HiltAndroidApp 将会触发 Hilt 的代码生成,作为程序依赖项容器的基类


生成的 Hilt 依附于 Application 的生命周期,他是 App 的父组件,提供访问其他组件的依赖


在 Application 中配置好后,就可以使用 Hilt 提供的组件了;组件包含 Application,Activity,Fragment,View,Service 等。


@HiltAndroidApp


创建一个依赖容器,该容器遵循 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 中的组件(Compenent)


使用 @Module 注解的类,需要使用 @Installin 注解来指定 module 的范围。


例如 @InstallIn(ApplicationComponent::class) 注解的 Module 就会绑定到 Application 的生命周期上。


Hilt 提供了以下组件来绑定依赖与对应 Android 类的活动范围

image.png

lt 没有为 broadcast receivers 提供组件,因为 Hilt 直接进从 ApplicationComponent 中注入 broadcast receivers。


Hilt 中组件的生命周期


Hilt 会根据相应的 Android 类生命周期自动创建和销毁组件的实例,对应关系如下:

image.png


如何使用 Hilt


buildscript {
    dependencies {
        //hilt
        classpath 'com.google.dagger:hilt-android-gradle-plugin:2.28-alpha'
    }
}


apply plugin: 'kotlin-kapt'
apply plugin: 'dagger.hilt.android.plugin'
//hilt
api "com.google.dagger:hilt-android:2.28-alpha"
kapt "com.google.dagger:hilt-android-compiler:2.28-alpha"


@HiltAndroidApp
class BaseApplication : Application() {
    override fun onCreate() {
        super.onCreate()
    }
}


到这里准备工作就做完了


使用 Hilt 进行依赖注入


class HiltTest @Inject constructor() {
    fun hiltTest() {
        Log.e("----------->", "hiltTest: ")
    }
}


@HiltAndroidApp
class BaseApplication : Application() {
    @Inject
    lateinit var hiltTest: HiltTest
    override fun onCreate() {
        super.onCreate()
        hiltTest.hiltTest()
    }
}


Hilt 在 Android 组件中的使用


如果使用 @AndroidEntryPoint 注解 Android 类,还必须注解依赖他的 Android 类;


例如: 给 fragment 使用 @AndroidEntryPoint 后,则还需要给 fragmet 依赖的 Activity 依赖 @AndroidEntryPoint ,否则会出现异常


@AndroidEntryPoint 不能以写在抽象类上


@AndroidEntryPoint 注解 仅仅支持 ComponentActivity 的子类,例如 Fragment,AppCompatActivity 等等。


@AndroidEntryPoint

class HomeNavigationActivity : BaseLayoutActivity<TestViewModel>() {
    override fun setViewModel(): Class<TestViewModel> =TestViewModel::class.java
    override fun layout(): Int {
        return R.layout.home_navigation
    }
    override fun bindView() {
    }
}


// fragment 中使用,需要本身所依赖的 activity 添加注解
@AndroidEntryPoint
class FragmentOne : BaseLayoutFragment<FragOneViewModel>() {
    //使用 @Inject 从组件中获取依赖进行注入
    @Inject
    lateinit var hiltTest: HiltTest
    override fun layout(): Int {
        return R.layout.frag_one
    }
    override fun bindView(rootView: View) {
        //对象已经注入,直接调用即可
        one.text = hiltTest.hiltTest()
    }
}


Hilt 和第三方组件的使用


如果需要在项目中注入第三方依赖,可以使用 @Module 注解。使用 @Module 在注解的普通类,在其中创建第三方依赖的对象即可。


@Module 模块用于向 Hilt 添加绑定,告诉 Hilt 如果提供不同类型的实例。


使用了 @Module 的类,相当于是一个模块,常用于创建依赖对象(如,Okhttp,Retrofit 等)。


使用 @Module 的类,需要使用 #InstallIn 指定此 module 的范围,会绑定到对应 Android 类的生命周期上


@Providers,常用于被 @Module 注解标记类的内部方法,并提供依赖项对象。


//对应的生命周期为 application
@Module
@InstallIn(ApplicationComponent::class)
object TestModule {
    /**
     * 每次都是新的实例
     */
    @Provides
    fun bindHiltTest(): HiltTest {
        XLog.e("--------bindHiltTest----")
        return HiltTest()
    }
    /**
     * 全局复用同一个实例
     */
    @Provides
    @Singleton
    fun bindSingTest(): Test {
        XLog.e("--------bindSingTest----")
        return Test()
    }
}


使用如下:


@Inject
lateinit var hiltTest: HiltTest
@Inject
lateinit var hiltTest1: 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-alpha01'

kapt 'androidx.hilt:hilt-compiler:1.0.0-alpha01'


通过 @ViewModelInject 注解进行构造注入。

SavedStateHandle 使用 @Asssisted 注解

class HomeContentViewModel @ViewModelInject  constructor(
    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() {
        launchVmHttp {
            liveData.postValue(response.requestBaidu())
        }
    }
}


通过 @Inject 进行注入,在 viewModel 中不需要手动的创建其对象

@ActivityScoped
class HomeContentRepository @Inject constructor() : BaseRepository() {
    suspend fun requestBaidu(): String {
        return LvHttp.createApi(ApiServices::class.java).baidu()
    }
}


获取 viewModel 的实例

@AndroidEntryPoint
class HomeContentActivity : AppCompatActivity(){
    //生成 ViewModel 的实例
    private val viewModel by viewModels<HomeContentViewModel>()
    override fun onCreate(savedInstanceState: Bundle?) {
        super.onCreate(savedInstanceState)
        setContentView(R.layout.activity_home_content)
        viewModel.requestBaiDu()
        viewModel.testLiveData.observe(this, Observer {
            ToastUtils.show(it)
        })
}


Hilt 和 Room 的使用


这里需要用到 @Module 注解,使用 @Module 注解的普通类,在其中提供 Room 的实例。并且使用 @InstallIn 来声明 作用范围。


@Module
@InstallIn(ApplicationComponent::class)
object RoomModel {
    /**
     * @Provides:常用于被 @Module 标记类的内部方法,并提供依赖对象
     * @Singleton:提供单例
     */
    @Provides
    @Singleton
    fun provideAppDataBase(application: Application): AppDataBase {
        return Room
            .databaseBuilder(application, AppDataBase::class.java, "knif.db")
            .fallbackToDestructiveMigration()
            .allowMainThreadQueries()
            .build()
    }
    @Provides
    @Singleton
    fun providerUserDao(appDataBase: AppDataBase): UserDao {
        return appDataBase.getUserDao()
    }
}


我们给 providerUserDao 使用了 @Provides 注解 和 @Singleton 注解,是为了告诉 Hilt,当使用 UserDao 时需要执行 appDataBase.getUserDao() 。


而在调用 appDataBase.getUserDao() 时需要传入 AppDataBase,这时就会调用上面的方法 provideAppDataBase 了,因为这个方法也是用了 @Provides 注解。


并且这两个方法都是单例,只会调用一次。


使用如下:


class FragmentTwo : BaseLayoutFragment<FragTwoViewModel>() {
    @Inject
    lateinit var userDao: UserDao
}


到现在为止,就可以在任意地方获取到 UserDao,并且不用手动的创建实例。


使用 @Binds 进行接口注入


Binds:必须注释一个抽象函数,抽象函数的返回值是实现的接口。通过添加具有接口实现类型的唯一参数来指定实现。


首先需要一个接口,和一个实现类


interface User {
    fun getName(): String
}


class UserImpl @Inject constructor() : User {
    override fun getName(): String {
        return "345"
    }
}


接着就需要新建一个 Module。用来实现接口的注入


@Module
@InstallIn(ApplicationComponent::class)
abstract class UserModule {
    @Binds
    abstract fun getUser(userImpl: UserImpl): User
}


注意:这个 Module 是抽象的。


使用如下:


@AndroidEntryPoint
class FragmentOne : BaseLayoutFragment<FragOneViewModel>() {
    @Inject
    lateinit var user: User
}


使用 @Qualifier 提供同一接口,不同的实现


还是上面的 User 接口,有两个不同的实现,如下:


class UserAImpl @Inject constructor() : User {
    override fun getName(): String {
        return "345"
    }
}


class UserBImpl @Inject constructor() : User {
    override fun getName(): String {
        return "Lv"
    }
}


接着定义两个注解


@Qualifier
annotation class A
@Qualifier
annotation class B


然后修改 Module ,在 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
}


这里用了两个不同的 mdule,并且对应两个不同的 component,一个是 application,另一个是 activity


最后使用如下:


@AndroidEntryPoint
class FragmentOne : BaseLayoutFragment<FragOneViewModel>() {
    @A
    @Inject
    lateinit var userA: User
    @B
    @Inject
    lateinit var userB: User
}


遇到的问题


在使用 @AndroidEntryPoint 注解的时候。需要在 fragment 和 actvity 都使用这个注解。


但是如果 activity 和 fragment 没在同一个module中,就会报错。


对于组件化的项目来说,这种情况就比较难受了。。。。


查找了一些资料:


主要问题之一是,通过在 Hilt 中发现模块的方式,无法区分哪些模块属于应用中的组件(如果他们确实使用过 Hilt) 已经库或其他库中的组件


另一个问题是,他将预先构建的组件层次结构变得复杂和混乱。就将你的库中所有活动一样,使父级成为 ApplicationComponent 也没有意义,因为您没有将组件放入 Application 。同样,如果一个仅包含片段库并托管在应用程序的活动中,那可能会遇到类似的情况,您希望库片段是独立的,单让 FragmentComponent 对象作为 ActivityComponent 并没有意义。


Hilt 好处


降低 Android 开发者使用依赖注入框架的上手成本

内部有一套标准的组件和作用域,对范围进行声明后,只能使用在指定的作用域中使用这个类,并且提供声明周期的管理,会自动释放不在使用的对象,减少资源的过度使用,提供代码的可重用性。

使用起来简单,告别繁琐的 new。。。 这种流程,只需要添加注解即可。提高了代码的可读性,构建简单,耦合变低,容易测试

我感觉最大的好处就是管理他们的生命周期,只能在对应的范围内进行使用。感觉非常好。


相关文章
|
1月前
|
编译器 Android开发 开发者
带你了解Android Jetpack库中的依赖注入框架:Hilt
本文介绍了Hilt,这是Google为Android开发的依赖注入框架,基于Dagger构建,旨在简化依赖注入过程。Hilt通过自动化的组件和注解减少了DI的样板代码,提高了应用的可测试性和可维护性。文章详细讲解了Hilt的主要概念、基本用法及原理,帮助开发者更好地理解和应用Hilt。
60 8
|
3月前
|
存储 移动开发 Android开发
使用kotlin Jetpack Compose框架开发安卓app, webview中h5如何访问手机存储上传文件
在Kotlin和Jetpack Compose中,集成WebView以支持HTML5页面访问手机存储及上传音频文件涉及关键步骤:1) 添加`READ_EXTERNAL_STORAGE`和`WRITE_EXTERNAL_STORAGE`权限,考虑Android 11的分区存储;2) 配置WebView允许JavaScript和文件访问,启用`javaScriptEnabled`、`allowFileAccess`等设置;3) HTML5页面使用`<input type="file">`让用户选择文件,利用File API;
|
4月前
|
JavaScript Java Android开发
kotlin安卓在Jetpack Compose 框架下跨组件通讯EventBus
**EventBus** 是一个Android事件总线库,简化组件间通信。要使用它,首先在Gradle中添加依赖`implementation &#39;org.greenrobot:eventbus:3.3.1&#39;`。然后,可选地定义事件类如`MessageEvent`。在活动或Fragment的`onCreate`中注册订阅者,在`onDestroy`中反注册。通过`@Subscribe`注解方法处理事件,如`onMessageEvent`。发送事件使用`EventBus.getDefault().post()`。
|
4月前
|
安全 JavaScript 前端开发
kotlin开发安卓app,JetPack Compose框架,给webview新增一个按钮,点击刷新网页
在Kotlin中开发Android应用,使用Jetpack Compose框架时,可以通过添加一个按钮到TopAppBar来实现WebView页面的刷新功能。按钮位于右上角,点击后调用`webViewState?.reload()`来刷新网页内容。以下是代码摘要:
|
4月前
|
缓存 Android开发 Kotlin
【安卓app开发】kotlin Jetpack Compose框架 | 先用OKhttp下载远程音频文件再使用ExoPlayer播放
使用 Kotlin 的 Jetpack Compose 开发安卓应用时,可以结合 OkHttp 下载远程音频文件和 ExoPlayer 进行播放。在 `build.gradle` 添加相关依赖后,示例代码展示了如何下载音频并用 ExoPlayer 播放。代码包括添加依赖、下载文件、播放文件及简单的 Compose UI。注意,示例未包含完整错误处理和资源释放,实际应用需补充这些内容。
|
4月前
|
JavaScript 前端开发 Android开发
kotlin安卓在Jetpack Compose 框架下使用webview , 网页中的JavaScript代码如何与native交互
在Jetpack Compose中使用Kotlin创建Webview组件,设置JavaScript交互:`@Composable`函数`ComposableWebView`加载网页并启用JavaScript。通过`addJavascriptInterface`添加`WebAppInterface`类,允许JavaScript调用Android方法如播放音频。当页面加载完成时,执行`onWebViewReady`回调。
|
Dart 前端开发 Java
用Jetpack Compose Desktop极简配置做一个Windows桌面时间显示器(compose框架入门向)
compose的模板配置多少有些臃肿,如果只做单一平台多少是会简单一些的。但几乎没怎么见过配置很简单的例子,都是套那些模板,我觉得没必要搞那么复杂,那么本文就做一个非常简单的只有几行代码的小例子
1675 0
用Jetpack Compose Desktop极简配置做一个Windows桌面时间显示器(compose框架入门向)
|
5月前
|
存储 安全 Android开发
构建高效的Android应用:Kotlin与Jetpack的结合
【5月更文挑战第31天】 在移动开发的世界中,Android 平台因其开放性和广泛的用户基础而备受开发者青睐。随着技术的进步和用户需求的不断升级,开发一个高效、流畅且易于维护的 Android 应用变得愈发重要。本文将探讨如何通过结合现代编程语言 Kotlin 和 Android Jetpack 组件来提升 Android 应用的性能和可维护性。我们将深入分析 Kotlin 语言的优势,探索 Jetpack 组件的核心功能,并通过实例演示如何在实际项目中应用这些技术。
|
4月前
|
数据管理 API 数据库
探索Android Jetpack:现代安卓开发的利器
Android Jetpack是谷歌为简化和优化安卓应用开发而推出的一套高级组件库。本文深入探讨了Jetpack的主要构成及其在应用开发中的实际运用,展示了如何通过使用这些工具来提升开发效率和应用性能。