Hilt是什么?
Hilt Google開源的一個(gè) Android 的依賴注入庫,其實(shí)是基于 Dagger。
Hilt 是專門為android打造的 ,可以使我們的代碼 盡量的簡化
?Hilt 創(chuàng)建了一組標(biāo)準(zhǔn)的 組件和作用域。這些組件會(huì)自動(dòng)集成到 Android 程序中的生命周期中。在使用的時(shí)候可以指定使用的范圍,事情作用在對(duì)應(yīng)的生命周期當(dāng)中。
Hilt如何使用?
(1)在項(xiàng)目的build.gradle中添加引用
dependencies {
...
classpath 'com.google.dagger:hilt-android-gradle-plugin:2.28-alpha'
}
}
(2)在App的build.gradle中添加plugin的引用的依賴
apply plugin: 'kotlin-kapt'
apply plugin: 'dagger.hilt.android.plugin'
dependencies {
implementation "com.google.dagger:hilt-android:2.28-alpha"
kapt "com.google.dagger:hilt-android-compiler:2.28-alpha"
}
Hilt的簡單用法
(1)hlit初始化
每個(gè) Android 程序中都會(huì)有一個(gè) Application,如果要是用hlit必須要自定義一個(gè) Application 才行,否則 Hilt 將無法正常工作。
@HiltAndroidApp
class BaseApplication : Application() {
override fun onCreate() {
super.onCreate()
}
}
(2)hlit支持的入口
hlit 一共支持6個(gè)入口點(diǎn)
- Application
- Activity
- Fragment
- View
- Service
- BroadcastReceiver
這些都是android的基礎(chǔ)控件 ,注意的是其中沒有contentProvider 因?yàn)閏ontentProvider的生命周期比較特殊,他是在Application的onCreate之前就已經(jīng)完成初始化了 ,而hlit實(shí)在Application中完成初始化的,所以自帶的入口點(diǎn)里面是沒有contentProvider的。
關(guān)于注解的聲明:只有 Application 這個(gè)入口點(diǎn)是使用 @HiltAndroidApp 注解來聲明的。其他的所有入口點(diǎn),都是用 @AndroidEntryPoint 注解來聲明的。
如果在一個(gè)activity中使用注解
@AndroidEntryPoint
class MainActivity : AppCompatActivity() {
override fun onCreate(savedInstanceState: Bundle?) {
super.onCreate(savedInstanceState)
setContentView(R.layout.activity_main)
}
}
注意點(diǎn):
如果使用 @AndroidEntryPoint 注解 Android 類,還必須注解依賴他的 Android 類;
例如: 給 fragment 使用 @AndroidEntryPoint 后,則還需要給 fragmet 依賴的 Activity 依賴@AndroidEntryPoint ,否則會(huì)出現(xiàn)異常
@AndroidEntryPoint 不能以寫在抽象類上
(3)hlit 完成簡單的實(shí)例化
- 聲明對(duì)象
class Test @Inject constructor(){
fun test() {
println("I am a Test test")
}
}
- 在添加注解的activity有中調(diào)用
@AndroidEntryPoint
class MainActivity : AppCompatActivity() {
@Inject
lateinit var test: Test
override fun onCreate(savedInstanceState: Bundle?) {
super.onCreate(savedInstanceState)
setContentView(R.layout.activity_main)
test.test()
}
}
-
輸出結(jié)果
image.png 注意
Hilt 注入的字段是不可以聲明成 private 的
*聲明需要參數(shù)的類
class Test @Inject constructor(var testPar: TestPar){
fun test() {
println("I am a Test test")
}
}
里面的參數(shù)同樣需要添加注解支持
class TestPar @Inject constructor(){
}
(4)hilt 完成接口實(shí)例化
- 聲明接口
interface Animal {
fun call()
}
- 編寫實(shí)現(xiàn)類 并使用 @Inject
class Cat @Inject constructor() : Animal {
override fun call() {
println("miao miao")
}
}
class Dog @Inject constructor() : Animal {
override fun call() {
println("wang wang")
}
}
- 聲明實(shí)現(xiàn)類的不同注解
@Qualifier
@Retention(AnnotationRetention.BINARY)
annotation class BindDog
@Qualifier
@Retention(AnnotationRetention.BINARY)
annotation class BindCat
- 編寫實(shí)現(xiàn)類接口對(duì)應(yīng)的module
@Module
@InstallIn(ActivityComponent::class)
abstract class AnimalModel {
@BindDog
@Binds
abstract fun bindDig(dog: Dog): Animal
@BindCat
@Binds
abstract fun bindCat(cat: Cat): Animal
}
- 在實(shí)現(xiàn)類中顯示
class Test @Inject constructor(var testPar: TestPar){
@BindDog
@Inject
lateinit var dog: Animal
@BindCat
@Inject
lateinit var cat: Animal
fun test() {
dog.call()
cat.call()
println("I am a Test test1")
}
}
-
最終輸出結(jié)果
image.png
(5)Hilt內(nèi)置組件和組件作用域
InstallIn,就是安裝到的意思。那么 @InstallIn(ActivityComponent::class),就是把這個(gè)模塊安裝到 Activity 組件當(dāng)中。同事這個(gè)也是對(duì)他的一種限制,表示只能在Activity 中只用。 Activity 中包含的 Fragment 和 View 也可以使用,但是除了 Activity、Fragment、View 之外的其他地方就無法使用了.
下面是一張簡單的作用域和范圍的表格

使用作用范圍表示在作用范圍內(nèi)是單例的。
(6)預(yù)置Qualifier
在android中與java不同的 android中打部分時(shí)候 頭需要傳遞一個(gè)比較重要的函數(shù)Context,但是現(xiàn)在Context 有沒有加上hilt對(duì)應(yīng)的注解 那么我們改如何在傳參是設(shè)置 Context呢?
這我們就用到了 hilt 預(yù)置的函數(shù)
預(yù)定義的綁定 - 表示 Android 類,如 Application 或 Activity。
預(yù)定義的限定符 - 表示 @ApplicationContext 和 @ActivityContext。
寫的時(shí)候這樣寫
class Test @Inject constructor(@ApplicationContext var context: Context){
@BindDog
@Inject
lateinit var dog: Animal
@BindCat
@Inject
lateinit var cat: Animal
fun test() {
dog.call()
cat.call()
println("I am a Test test1")
}
}
這樣就可以添加Context關(guān)鍵字了

