Hilt使用姿势全解析

Hilt是什么?

Hilt是Android的依赖注入库,可以减少在项目中执行手动依赖项注入的样板代码。执行手动依赖项注入需要手动构造每个类及其依赖项,并借助容器重复使用和管理依赖项。

和Degger有什么不同?

Hilt是在Degger的基础上构建而成,更具场景化。Hilt通过为项目中的每个Android类提供容器并自动管理其生命周期,在简化依赖注入使用的同时保留了Degger原有的强大功能。(类似于retrofit与okhttp的关系)

Hilt凭什么吸引我?

是数据共享和依赖管理,确切一点是生命周期内数据共享,是不是有点不知所云,没关系咱看代码
一个数据类:

@ActivityScoped
public class User {
    private String name;
    private String mood;

    @Inject
    public User() {
        this("matthew", "百无聊赖");
    }

    public User(String name, String mood) {
        this.name = name;
        this.mood = mood;
    }
    ......省略不重要的代码
}

Activity

@AndroidEntryPoint
class MainActivity : AppCompatActivity() {
    @Inject
    lateinit var user: User

    override fun onCreate(savedInstanceState: Bundle?) {
        super.onCreate(savedInstanceState)
        setContentView(R.layout.activity_main)

        user.mood = "一切都好"
     }
 ......省略不重要的代码
}

一个自定义view继承于AppCompatTextView ,它需要使用User对象的数据进行展示.它被放在了MainActivity的layout里

@AndroidEntryPoint
public class UserView extends androidx.appcompat.widget.AppCompatTextView {
    @Inject
    User user;
     ......省略不重要的代码
    @Override
    protected void onAttachedToWindow() {
        super.onAttachedToWindow();
        setText(user.getName() + "现在的心情是" + user.getMood());
    }
}

我们的使用场景是,需要在MainActivity里动态的去修改User里的值,然后UserView去展示修改过后的值。老司机门一定马上就想到了,先在UserView添加一个接收User对象的方法,比如public void setUser(User user){...}然后在activity里获取userView的对象并调用userView.setUser函数把对象传过去。这种方式我用了好多年了,但是接触hilt之后瞬间感觉不优雅了。假设之后需求变了不需要这个User对象了呢?。。。。一顿删除。 。。。。头都大了吧
如果使用Hilt来依赖注入,我们只需要删除注入的地方@Inject User user;即可,就是这么简单。
例子举的还是有一点点不太理想

想在介绍一大堆枯燥的规则配置项之前先给大家讲讲怎么最简单的实现上面的生命周期类共享数据(只讲操作,为什么这么写后面会讲)
1、添加依赖项 (往下翻翻,不重复写了)
2、给Applicaton添加@HiltAndroidApp注解

@HiltAndroidApp
public class ExampleApplication extends Application { ... }

3、准备需要注入的数据类,给无参构造函数添加@Inject注解,给User类添加@ActivityScoped注解

@ActivityScoped
public class User {
    private String name;
    private String mood;

    @Inject
    public User() {
        this("matthew", "百无聊赖");
    }

    public User(String name, String mood) {
        this.name = name;
        this.mood = mood;
    }
    ......省略不重要的代码
}

4、需要时引入
给需要依赖注入的类添加@AndroidEntryPoint注解
给对象声明添加@Inject注解

@AndroidEntryPoint
class MainActivity : AppCompatActivity() {
    @Inject
    lateinit var user: User

    override fun onCreate(savedInstanceState: Bundle?) {
        super.onCreate(savedInstanceState)
        setContentView(R.layout.activity_main)

        user.mood = "一切都好"
     }
 ......省略不重要的代码
}
@AndroidEntryPoint
public class UserView extends androidx.appcompat.widget.AppCompatTextView {
    @Inject
    User user;
     ......省略不重要的代码
    @Override
    protected void onAttachedToWindow() {
        super.onAttachedToWindow();
        setText(user.getName() + "现在的心情是" + user.getMood());
    }
}

至此就完成了!!!

Hilt怎么用

  • 添加依赖项

首先,将 hilt-android-gradle-plugin 插件添加到项目的根级 build.gradle 文件中:

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

然后,应用 Gradle 插件并在 app/build.gradle 文件中添加以下依赖项:

...
apply plugin: 'kotlin-kapt'
apply plugin: 'dagger.hilt.android.plugin'

android {
    ...
}

dependencies {
    implementation "com.google.dagger:hilt-android:2.28-alpha"
    kapt "com.google.dagger:hilt-android-compiler:2.28-alpha"
}

Hilt 使用 Java 8 功能。如需在项目中启用 Java 8,请将以下代码添加到 app/build.gradle 文件中:

android {
  ...
  compileOptions {
    sourceCompatibility JavaVersion.VERSION_1_8
    targetCompatibility JavaVersion.VERSION_1_8
  }
}
  • Hilt 目前支持以下 Android 类:

通过使用 @HiltAndroidApp
Application()
其它通过使用@AndroidEntryPoint
Activity
Fragment
View
Service
BroadcastReceiver

@HiltAndroidApp 会触发 Hilt 的代码生成操作,生成的代码包括应用的一个基类,该基类充当应用级依赖项容器。

@HiltAndroidApp
public class ExampleApplication extends Application { ... }

在 Application 类中设置了 Hilt 且有了应用级组件后,Hilt 可以为带有 @AndroidEntryPoint 注释的其他 Android 类提供依赖项:

@AndroidEntryPoint
public class ExampleActivity extends AppCompatActivity { ... }
  • 注入

如需从组件获取依赖项,请使用 @Inject 注释执行字段注入:

@AndroidEntryPoint
public class ExampleActivity extends AppCompatActivity {

  @Inject
  AnalyticsAdapter analytics;
  ...
}
  • 定义Hilt绑定

为了执行字段注入,Hilt需要知道如何从相应组件提供必要依赖项的实例,向Hilt提供绑定信息的一种方法是构造函数注入。在类的构造函数中使用@Inject注释以告知Hilt如何提供该类实例。

public class AnalyticsAdapter {

  private final AnalyticsService service;

  @Inject
  AnalyticsAdapter(AnalyticsService service) {
    this.service = service;
  }
  ...
}
  • Hilt 模块

上例中给出的构造函数需要一个参数,那么我们还需要告知Hilt如何获得AnalyticsService的实例。很简单啊!对AnalyticsService的构造函数添加@Inject不就可以了。
但是有时候实例不能通过构造函数注入。发生这种情况可能有多种原因。例如,不能通过构造函数注入接口;或者不归我们所有的类型,比如第三方库;或者必须通过构造器构建的实例。
怎么办?需要用到Hilt模块,一种带有@Module注释的类。它会告诉Hilt如何提供某些类型的实例。同时必须使用@InstallIn为模块添加注释,告诉Hilt每个模块将用在或安装在那个Android类中。

  • 使用 @Provides 注入实例

如果类不归我们所有,或者需要使用构造器时,可以在Hilt模块内创建一个函数添加 @Provides注解
@Provides注解会告诉Hilt以下信息
1、函数返回类型会告诉Hilt提供哪个类型的实例
2、函数参数或告诉Hilt相应类型的依赖项
3、函数体会告诉Hilt如何提供相应类型的实例

@Module
@InstallIn(ActivityComponent.class)
public class AnalyticsModule {

  @Provides
  @ActivityScoped  //共享范围
  public static AnalyticsService provideAnalyticsService(
    // Potential dependencies of this type
  ) {
      return new Retrofit.Builder()
               .baseUrl("https://example.com")
               .build()
               .create(AnalyticsService.class);
  }
}
  • 使用 @Binds 注入接口实例

如果 AnalyticsService是一个接口,则无法通过构造函数注入,而应向Hilt提供绑定信息,在Hilt模块内创建一个带有@Binds 注释的抽象函数。
带有注释的函数会向Hilt提供以下信息
1、函数返回类型会告诉Hilt需要提供哪个接口的实例
2、函数参数会告诉Hilt要提供哪种实现

public interface AnalyticsService {
  void analyticsMethods();
}

// Constructor-injected, because Hilt needs to know how to
// provide instances of AnalyticsServiceImpl, too.
public class AnalyticsServiceImpl implements AnalyticsService {
  ...
  @Inject
  AnalyticsServiceImpl(...) {
    ...
  }
}

@Module
@InstallIn(ActivityComponent.class)
public abstract class AnalyticsModule {

  @Binds
  public abstract AnalyticsService bindAnalyticsService(
    AnalyticsServiceImpl analyticsServiceImpl
  );
}
  • Hilt组件

可以从中执行字段注入的每个Android类都有一个与之关联的Hilt组件。可以在@InstallIn中引入该组件,每个Hilt负责将其绑定注入相应的Android类。

Hilt 组件 注入器面向的对象
ApplicationComponent Application
ActivityRetainedComponent ViewModel
ActivityComponent Activity
FragmentComponent Fragment
ViewComponent View
ViewWithFragmentComponent 带有 @WithFragmentBindings 注释的 View
ServiceComponent Service

那么比如使用@InstallIn(ActivityComponent.class)将module注入到activity之后,activity下的view能否获得依赖呢?具体规则是什么?

  • 组件层级结构

将模块安装到组件后,其绑定就可以用做该组件内其它绑定的依赖项,也可以用作组件层级中该组件下的任何子组件中其它绑定的依赖项。就是向下使用关系


hilt.png

注意:默认情况下,如果您在视图中执行字段注入,ViewComponent 可以使用 ActivityComponent 中定义的绑定。如果您还需要使用 FragmentComponent 中定义的绑定并且视图是 Fragment 的一部分,应将 @WithFragmentBindings 注释和 @AndroidEntryPoint 一起使用。

  • 组件的声明周期

Hilt会根据Android类的生命周期自动创建和销毁组件类的实例

生成的组件 创建时机 销毁时机
ApplicationComponent Application#onCreate() Application#onDestroy()
ActivityRetainedComponent Activity#onCreate() Activity#onDestroy()
ActivityComponent Activity#onCreate() Activity#onDestroy()
FragmentComponent Fragment#onAttach() Fragment#onDestroy()
ViewComponent View#super() 视图销毁时
ViewWithFragmentComponent View#super() 视图销毁时
ServiceComponent Service#onCreate() Service#onDestroy()

注意:ActivityRetainedComponent 在配置更改后仍然存在,因此它在第一次调用 Activity#onCreate() 时创建,在最后一次调用 Activity#onDestroy() 时销毁。

  • 作用域

如果使用@ActivityScoped将AnalyticsService的作用域限定为ActivityComponent,Hilt会在Activity的整个周期内提供AnalyticsService的同一实例

@Provides
  @ActivityScoped  //共享范围
  public static AnalyticsService provideAnalyticsService(
    // Potential dependencies of this type
  ) {
      return new Retrofit.Builder()
               .baseUrl("https://example.com")
               .build()
               .create(AnalyticsService.class);
  }
Android 类 生成的组件 作用域
Application ApplicationComponent @Singleton
View Model ActivityRetainedComponent @ActivityRetainedScope
Activity ActivityComponent @ActivityScoped
Fragment FragmentComponent @FragmentScoped
View ViewComponent @ViewScoped
带有 @WithFragmentBindings 注释的 View ViewWithFragmentComponent @ViewScoped
Service ServiceComponent @ServiceScoped
  • Hilt中的预定义限定符

Hilt提供了一些自定义的限定符,例如开发中可能需要来自应用或者Activity的context对象。因此 Hilt 提供了 @ApplicationContext@ActivityContext限定符。

public class AnalyticsAdapter {

  private final Context context;
  private final AnalyticsService service;

  @Inject
  AnalyticsAdapter(
    @ActivityContext Context context,
    AnalyticsService service
  ) {
    this.context = context;
    this.service = service;
  }
}
  • 为同一类型提供多个绑定

我将这个放在了最后,为什么?因为按照我的使用习惯,这个用到的情况比较少。
比如现在需要获取同一个接口的不同实例。上面讲到了使用 @Binds 注入接口实例,但是现在需要多个不同的实例,该怎么办?返回值类型都是一样的啊!!!
首先,要定义用于@Binds@Provides方法添加注释的限定符

@Qualifier
@Retention(RetentionPolicy.RUNTIME)
private @interface AuthInterceptorOkHttpClient {}

@Qualifier
@Retention(RetentionPolicy.RUNTIME)
private @interface OtherInterceptorOkHttpClient {}

然后,Hilt需要知道如何提供与每个限定符对应的类型的实例。下面的两个方法具有相同的返回值类型,但是限定符将它们标记为两个不同的绑定。

@Module
@InstallIn(ActivityComponent.class)
public class NetworkModule {

  @AuthInterceptorOkHttpClient
  @Provides
  public static OkHttpClient provideAuthInterceptorOkHttpClient(
    AuthInterceptor authInterceptor
  ) {
      return new OkHttpClient.Builder()
                   .addInterceptor(authInterceptor)
                   .build();
  }

  @OtherInterceptorOkHttpClient
  @Provides
  public static OkHttpClient provideOtherInterceptorOkHttpClient(
    OtherInterceptor otherInterceptor
  ) {
      return new OkHttpClient.Builder()
                   .addInterceptor(otherInterceptor)
                   .build();
  }
}

在使用过程中使用相应的限定符为字段或参数添加注释来注入所需的特定类型:

// As a dependency of another class.
@Module
@InstallIn(ActivityComponent.class)
public class AnalyticsModule {

  @Provides
  public static AnalyticsService provideAnalyticsService(
    @AuthInterceptorOkHttpClient OkHttpClient okHttpClient
  ) {
      return new Retrofit.Builder()
                  .baseUrl("https://example.com")
                  .client(okHttpClient)
                  .build()
                  .create(AnalyticsService.class);
  }
}

// As a dependency of a constructor-injected class.
public class ExampleServiceImpl ... {

  private final OkHttpClient okHttpClient;

  @Inject
  ExampleServiceImpl(@AuthInterceptorOkHttpClient OkHttpClient okHttpClient) {
    this.okHttpClient = okHttpClient;
  }
}

// At field injection.
@AndroidEntryPoint
public class ExampleActivity extends AppCompatActivity {

  @AuthInterceptorOkHttpClient
  @Inject
  OkHttpClient okHttpClient;
  ...
}
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 215,245评论 6 497
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,749评论 3 391
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 160,960评论 0 350
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,575评论 1 288
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,668评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,670评论 1 294
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,664评论 3 415
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,422评论 0 270
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,864评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,178评论 2 331
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,340评论 1 344
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,015评论 5 340
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,646评论 3 323
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,265评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,494评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,261评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,206评论 2 352