Dagger 在 android 中的使用

依赖

  • 什么是依赖?举个例子:当你写一个Activity 你需要把一些数据写入数据库,这时你一般会再写个 DatabaseMgr 来实现这个功能能。 这样你的这个 Activity 将依赖 DatabaseMgr 来实现读写数据库的功能, DatabaseMgr 也就成为了 Activity 的依赖。

  • 获取依赖的3种方式

    1. The class constructs the dependency it needs.
    2. Grab it from somewhere else. Some Android APIs, such as Context getters and getSystemService(), work this way.
    3. Have it supplied as a parameter.

    The third option is dependency injection! With this approach you take the dependencies of a class and provide them rather than having the class instance obtain them itself.

  • 既然依赖以参数形式注入,那么构建一个依赖链路很深的对象就特别麻烦,Dagger 为我们提的自动构建依赖。

  • 我们需要通过各种注解告诉 Dagger 三种信息:

    1. 谁依赖谁?通过 @Component 定义依赖图,Dagger 以这个为根节点,递归解析从它的依赖,它的依赖的依赖, 它依赖的依赖的依赖... 并生成代码来构建这些依赖。

      A @Component interface gives the information Dagger needs to generate the graph at compile-time. The parameter of the interface methods define what classes request injection.
      
    2. 如何构建依赖?有2种方式:

      • 通过@Inject 注解依赖的构造方法;

        class LoginViewModel @Inject constructor() { ... }
        
      • @Module 里实现依赖的构建,并用 @Provider 或者 @Binds 注解;

        @Module
        class LoginModule {
        
            @Provides
            fun getLocalData(): LocalData{
                return LocalData()
            }
        }
        
        @Module
        abstract class StorageModule {
        
            @Binds
            abstract fun provideStorage(storage: SharedPreferencesStorage): Storage
        }
        

        @Binds 其实本身并不构建依赖,只是在需要构建一个接口的实现时,告诉 Dagger 用特定的某个实现罢了;

    3. 如何管理依赖对象的生命周期? Use @Scope to scope a type to the Component's lifecycle. 如果这个依赖是通过 @Provider 创建的,那么要注解 @Provider 的方法,而不是这个依赖

@Component@Module@Subcomponent 之间的关系

  • Component 是个接口或者抽象类,它的生命周期一般和 Application 对应,Dagger 会生成代码实现它,我们就可以用这个类来获取(或者注入)依赖,或者获取它的 Subcomponent 来获取(或者注入)依赖;

  • Module 是普通或者抽象类,用来给它所属的 Component/Subcomponent 提供特殊依赖的构建;

    Similar to Components, Dagger Modules tell Dagger how to provide instances of a certain type. Dependencies are defined using the @Provides and @Binds annotations.
    
  • SubcomponentComponent 的功能类似,只是它从属于 Component ,必须通过 Component 来使用它。 Dagger 在编译时会生成 Subcomponent 的实现并作为 Component 的实现类的内部类,因此 Subcomponent 可以获取到 Component 所提供其他依赖;

  • SubComponent 不能直接挂在Component 上, 需要定义个 Module 承载它,然后再把Module 挂到 Component上:

    @Subcomponent
    interface LoginComponent { ... }
    
    
    @Module(subcomponents = LoginComponent::class)
    class SubcomponentsModule { ... }
    
    @Component(modules = [SubcomponentsModule::class])
    interface ApplicationComponent { ... }
    
  • Module 一般在同一个 Component 里面只定义一次;

    Good practice dictates that modules should only be declared once in a component (outside of specific advanced Dagger use cases).

Dagger 生成代码

dagger_generated_code.png
  • Dagger 为 Component 生成 Dagger$Component 类,该类实例通过静态方法 create 以 Builder 的方式创建;

  • Dagger$Component 的成员变量有2类:

    • @Module 注解的类, 因为这种类定义了如何获取依赖,所以当然得持有它;
    • Provider, 用来生产依赖,如果依赖生命周期和当前 Component 相同, 就用 DoubleCheck 做缓存,这样每次就不会生产新的依赖;
  • Dagger$Component 有3种内部类

    • Subcomponent 的实现类,内部类可以访问外部类的成员变量,因此 Subcomponent 可以获取到所有 Component 可以获取到的依赖。

    • 其他2种分别是用以创建自己的 Factory 工厂类或者 Builder 类,这2种方式都是用来创建 Dagger$Component 自身,因此没有必要2者都同时存在,有了 Factory 就不会有 BuilderFactoryBuilder 可以通过 @Component.Factory@Component.Builder 来显示定义。对于 Component 一般可以不用显示定义它的 Factory或者Builder,但 SubComponent 一般都需要显示定义它的Factory; 原因如下:

      • 因为使用者不能直接使用Subcomponent,使用者必须通过 Dagger$Component 来获取它的实例,而 Dagger$Component 获取 Subcomponent 的实例也是通过 SubcomponentFactory(或Builder),如果 Subcomponent 没有显示定义, Component 不清楚隐式的(也就是将要生成的默认的)FactoryBuilder 具体是什么样的,也就不好生成对应的代码。(其实我认为 Dagger 应该是要知道的,因为代码都是它自己生成的)
      • 使用者要通过 Component 获取 Subcomponent 的实例,就需要在 Component 里定义好获取 Subcomponent 的实例接口或者生成 SubcomponentFactory(Builder), 对于前者,缺乏可扩展性,所以 Dagger 选择了后者。 但 SubcomponentFactoryBuilder 将会是什么样的? 一种情况是隐式生成的,这就要在生成完隐式的 Factory 代码然后把它定义在 Component 里面,但 Subcomponent 代码生成又依赖 Component 的代码生成,代码生成过程会出现互相依赖的死循环。另外一种情况是显式定义的 Factory, 这样就可以避免死循环依赖了,所以需要显示定义 SubcomponentFactory 或者 Builder
  • Subcomponent 的生成代码,将和 Component 生成的代码类似,这样就可以实现模块之间的嵌套;

Dagger For Android

  • 在 app 中,依赖根节点一般是 ApplicationActivityFragment,比如在Activity 里面有个依赖需要注入,而这些对象是在 Framework 层构建的,所以这些对象的依赖需要手动调用 Dagger 来实现,比如在 Application#onCreate, Activity#onCreate, Fragment#onAttach ;

    在 Activity 或者 Fragment 比较多的情况下,会重复很多类似下面这样的代码片段:

    (application as MyApplication).appComponent.loginComponent().create().inject(this)
    

    这样的代码有几个问题:

    • 重复
    • 依赖使用者需要知道依赖是怎么注入的,通过 applicaiton 转换为 MyApplication 在拿到 loginComponent 的 Factory 创建一个 Subcomponent,然后调用它的 inject 方法。 这样一系列的操作,对依赖使用者来说太不透明了。

    那有没有办法定义一个 BaseActivity 或者 BaseFragment,在这里执行一次注入,并且注入过程对于使用者来说是透明的呢?那就要用到 AndroidInjector ,先说一下使用方式,然后再解释一下原理:

使用方法

总共有 4 个步骤

  1. Application 类实现 HasAndroidInjector接口:

    class MyApplication: Application(), HasAndroidInjector{
    
        @Inject
        lateinit var dispatchingAndroidInjector: DispatchingAndroidInjector<Any>
    
        override fun androidInjector(): AndroidInjector<Any> {
            return dispatchingAndroidInjector;
        }
    
        override fun onCreate() {
            super.onCreate()
            DaggerApplicationComponent.create().inject(this)
        }
    }
    
  2. Component include 内置的 AndroidInjectionModule

    @Component(modules = [AndroidInjectionModule::class, ...]) 
        interface ApplicationComponent {
        ...
    }
    
  3. Activity 定义一个 Subcomponent:

  @Subcomponent
  interface LoginComponent2: AndroidInjector<LoginActivity> {
  
      @Subcomponent.Factory
      interface Factory: AndroidInjector.Factory<LoginActivity> {}
  }

  @Module(subcomponents = [LoginComponent2::class])
  abstract class LoginModule2{
  
      @Binds
      @IntoMap
      @ClassKey(LoginActivity::class)
      abstract fun provideLoginComponent2Factory(f: LoginComponent2.Factory): AndroidInjector.Factory<*>
  }
  1. LoginModule2 include 到 Component中后,就可以在 BaseActivity 中注入它的子类了:

    abstract class BaseActivity: Activity() {
    
        override fun onCreate(savedInstanceState: Bundle?) {
            AndroidInjection.inject(this)
            super.onCreate(savedInstanceState)
        }
    }
    

    以后新加的 Activity ,只要它继承 BaseActivity , 并定义好它自己的 SubcomponentModule ,就可以实现依赖的自动注入了;

第 3 步骤,给每个 Activity 定义 SubcomponentModule 还是比较繁琐的,用 @ContributesAndroidInjector 就可以让 Dagger 自动为我们生成 SubcomponentModule:

@Module
abstract class ActivityModule {
    
    @ContributesAndroidInjector
    abstract fun loginActivity(): LoginActivity
}

Dagger 会生成如下代码(基本和手动写的差不多)

@Module(subcomponents = ActivityModule_LoginActivity.LoginActivitySubcomponent.class)
public abstract class ActivityModule_LoginActivity {
  private ActivityModule_LoginActivity() {}
    
  @Binds
  @IntoMap
  @ClassKey(LoginActivity.class)
  abstract AndroidInjector.Factory<?> bindAndroidInjectorFactory(
      LoginActivitySubcomponent.Factory builder);
    
  @Subcomponent
  public interface LoginActivitySubcomponent extends AndroidInjector<LoginActivity> {
    @Subcomponent.Factory
    interface Factory extends AndroidInjector.Factory<LoginActivity> {}
  }
}
原理
  • 通过 @IntoMap @ClassKey 注解,告诉 Dagger 用 LoginComponent2.Factory 这个 Factory 生成的 Subcomponent 来注入 LoginActivity。 这样生成的 Dagger$Component 代码会建立一个 Activity 对应 Provider 的 map:

    private Map<Class<?>, Provider<AndroidInjector.Factory<?>>> getMapOfClassOfAndProviderOfAndroidInjectorFactoryOf() {
        return Collections.<Class<?>, Provider<AndroidInjector.Factory<?>>>singletonMap(LoginActivity.class, (Provider) loginActivitySubcomponentFactoryProvider);
    }
    
  • 这个 map 会被传入 DispatchingAndroidInjector 里面:

    public final class DispatchingAndroidInjector<T> implements AndroidInjector<T> {
        ...
        private final Map<String, Provider<AndroidInjector.Factory<?>>> injectorFactories;
        ...
    }
    
  • 然后 AndroidInjector.inject 其实就是通过DispatchingAndroidInjector.inject来实现注入的:

    public boolean maybeInject(T instance) {
    Provider<AndroidInjector.Factory<?>> factoryProvider =
        injectorFactories.get(instance.getClass().getName());
    
        AndroidInjector.Factory<T> factory = (AndroidInjector.Factory<T>) factoryProvider.get();
        ...
        AndroidInjector<T> injector = checkNotNull(
              factory.create(instance), "%s.create(I) should not return null.", factory.getClass());
    
        injector.inject(instance);
    }
    
使用过程注意点
  • dagger、dagger-android-support 等各个插件版本要保持一致:

    implementation 'com.google.dagger:dagger:2.27'
    kapt 'com.google.dagger:dagger-compiler:2.27'
    
    annotationProcessor 'com.google.dagger:dagger-compiler:2.27'
    implementation 'com.google.dagger:dagger-android-support:2.27'
    kapt "com.google.dagger:dagger-android-processor:2.27"
    

Disadvantages of Dagger

  • 学习成本高;
  • 进一步限定了app的开发模式;
  • 需要暴露依赖成员变量的访问属性;
  • 阅读代码不方便,依赖和使用者隔离,逻辑不直观连贯;

学习资料

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

推荐阅读更多精彩内容