dagger2从入门到放弃-ActivityMultibindings

前面文章中讲到Componnet继承和依赖的三种方式时说到了在父Componnet中Module中指定SubComponent,这种方式主要用来实现Activity-Multibindings,现在就来了解下什么是Activity-Multibindings

先说说什么是Multibindings

简单的说就是将多个对象放到一个集合中,让依赖需求方可以通过一个集合找到需要的依赖而不用绑定具体的依赖

Set multibindings

在@Module中使用 @IntoSet或者 @ElementsIntoSet注解

@Module
class MyModuleA {
  //提供一个元素  
  @Provides @IntoSet
  static String provideOneString(DepA depA, DepB depB) {
    return "ABC";
  }
}

@Module
class MyModuleB {
  //一次提供多个元素  
  @Provides @ElementsIntoSet
  static Set<String> provideSomeStrings(DepA depA, DepB depB) {
    return new HashSet<String>(Arrays.asList("DEF", "GHI"));
  }
}

//dagger可以提供一个Set<String>
class Bar {
  @Inject Bar(Set<String> strings) {
    assert strings.contains("ABC");
    assert strings.contains("DEF");
    assert strings.contains("GHI");
  }
}

//或者从Componnet中获取Set<String>
@Component(modules = {MyModuleA.class, MyModuleB.class})
interface MyComponent {
  Set<String> strings();
}

@Test void testMyComponent() {
  MyComponent myComponent = DaggerMyComponent.create();
  assertThat(myComponent.strings()).containsExactly("ABC", "DEF", "GHI");
}

Map multibindings

只介绍最简单的方式,里面用到@IntoMap(标记依赖是要添加到Map中)和@MapKey(定义Map的key)

@Module
class MyModule {
  @Provides @IntoMap
  @StringKey("foo")
  static Long provideFooValue() {
    return 100L;
  }

  @Provides @IntoMap
  @ClassKey(Thing.class)
  static String provideThingValue() {
    return "value for Thing";
  }
}

@Component(modules = MyModule.class)
interface MyComponent {
  Map<String, Long> longsByString();
  Map<Class<?>, String> stringsByClass();
}

@Test void testMyComponent() {
  MyComponent myComponent = DaggerMyComponent.create();
  assertThat(myComponent.longsByString().get("foo")).isEqualTo(100L);
  assertThat(myComponent.stringsByClass().get(Thing.class))
      .isEqualTo("value for Thing");
}

还有其他复杂点的用法

  • 使用enum作为MapKey
  • 使用复杂的MapKey
  • 将set转换为map的用法
  • SubComponent将元素绑定到父Componnet中的集合中

具体见官方文档multibindings

Activities Subcomponents Multibinding

之前注入的过程基本是这样

getAppCompoent().getActivityComponent2().inject(this);

DaggerActivityComponent1.builder()
    .appComponent(getAppCompoent())
    .build().inject(this);

都需要获取到父Componnet的实例才能完成注入,也就是被注入的Activity需要了解注射器是如何组织的,这不符合依赖注入的核心原则:一个类不应该知道如何实现依赖注入

使用Activities Subcomponents Multibinding可以解决上述问题

先看看如何使用

step1

先定义一个ActivityInjector接口,可以用来完成对Activity的注入

public interface ActivityInjector<A extends Activity> extends MembersInjector<A> {
}

step2

定义一个ActivityModule,用来传入一个activity实例并作为依赖提供

@Module
public abstract class ActivityModule<T> {
    protected final T activity;

    public ActivityModule(T activity) {
        this.activity = activity;
    }

    @Provides
    @ActivityScope
    public T provideActivity() {
        return activity;
    }
}

step3

定义通用的ActivityComponentBuilder接口,用来传入ActivityModule并构建ActivityInjector实例

public interface ActivityComponentBuilder<M extends ActivityModule, C extends ActivityInjector> {
    ActivityComponentBuilder<M, C> activityModule(M activityModule);

    C build();
}

step4

定义注入Activity的SubComponent

@ActivityScope
@Subcomponent(modules = ActivityComponent3.SubcomponentActivityModule.class)
public interface ActivityComponent3 extends ActivityInjector<SubComponentActivity3> {


    @Subcomponent.Builder
    interface Builder extends ActivityComponentBuilder<SubcomponentActivityModule, ActivityComponent3> {
    }

    @Module
    class SubcomponentActivityModule extends ActivityModule<SubComponentActivity3> {
        public SubcomponentActivityModule(SubComponentActivity3 activity) {
            super(activity);
        }
    }
}

step5

定义一个module,将ActivityComponentBuilder作为一个元素添加到Map中,使用Activity的类型作为MapKey

@Module(subcomponents = ActivityComponent3.class)
public abstract class ActivitisMultibindingModule {
    @Binds
    @IntoMap
    @ActivityKey(SubComponentActivity3.class)
    public abstract ActivityComponentBuilder
    bindSubComponentActivityBuilder(ActivityComponent3.Builder builder);
}

step6

定义HasActivityComponentBuilder接口,传入MapKey获取ActivityComponentBuilder,这里的MapKey是Activity的Class类型

public interface HasActivityComponentBuilder {
    ActivityComponentBuilder getActivityComponentBuilder(Class<? extends Activity> activityClass);
}

step7

Application中实现HasActivityComponentBuilder接口,从注入到Application中的Map<Class<? extends Activity>, Provider<ActivityComponentBuilder>>以MapKey获取ActivityComponentBuilder

public class RealApplication extends BaseApplication implements HasActivityInjector, HasActivityComponentBuilder {
    private static final String TAG = "RealApplication";

    @Inject
    Map<Class<? extends Activity>, Provider<ActivityComponentBuilder>> activityComponentBuilders;

    //只是为了拿到Application
    public static HasActivityComponentBuilder get(Context context) {
        return ((HasActivityComponentBuilder) context.getApplicationContext());
    }
    
    //用activityClass去找对应的ActivityComponentBuilder
    @Override
    public ActivityComponentBuilder getActivityComponentBuilder(Class<? extends Activity> activityClass) {
        return activityComponentBuilders.get(activityClass).get();
    }
}

step8

在Activity中完成注入

    private void injectMembers() {
        RealApplication.get(this).getActivityComponentBuilder(SubComponentActivity3.class)
                .activityModule(new ActivityComponent3.SubcomponentActivityModule(this))
                .build()
                .injectMembers(this);
    }

个人理解的Activities Subcomponents Multibinding

实质上是将Activity的Subcomponnet/Subcomponent.Builder作为Map的value,使用时不从最底层的Component一级级的构建Component,而是用自身的Class作为key来获取Subcomponnet/Subcomponent.Builder,最终完成注入

优点

可以看到在最后一步的注入过程中,Activity只是获取Application-传入自身作为Module的参数,完成注入;整个过程都没有Componnet的身影,这就是Activities Subcomponents Multibinding的好处了

而且以上步骤不仅适用于Activity,对Fragment,View,ViewModel等都可以用同样的套路去屏蔽上层对如何完成注入的了解

缺点

当然有得有舍

  • 整个实现过程多了很多步骤
  • ActivityComponentBuilder是一个接口,导致只能将当前Activity实例作为参数,想要扩展其他参数作为依赖就没有办法了

对于第一个缺点,仅仅为了不去了解依赖注入框架的结构就多写那么多代码,有些得不偿失,不过所幸这些都是套路,满眼都是模板代码,所以google推出了dagger.android来简化android中四大组件以及Fragment中的Subcomponents Multibinding

下一章就来介绍dagger.android

参考资料

[Android]在Dagger 2中Activities和Subcomponents的多绑定(翻译)

相关文章

dagger2从入门到放弃-概念
dagger2从入门到放弃-最基础的用法介绍
dagger2从入门到放弃-Component的继承体系、局部单例
dagger2从入门到放弃-ActivityMultibindings
dagger2从入门到放弃-dagger.android
dagger2从入门到放弃-其他用法
dagger2从入门到放弃-多模块项目下dagger的使用
dagger2从入门到放弃-为何放弃

示例代码

DaggerInAction
欢迎star
master分支上最新的代码可能会比当前文章的示例代码稍微复杂点,提交记录里包含了每一步的迭代过程,可以顺藤摸瓜

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

推荐阅读更多精彩内容