dagger2 学习(二)

前言

之前记录了最简单的Dagger 2 使用,现在记录一下面对多层依赖时的问题,同时配合 @Module 进行注入的情况。

A 多层依赖情况

该部分代码A

1. 多层依赖情况模拟

添加 ClassRoom

public class ClassRoom {

    User mUser;

    @Inject
    public ClassRoom(User user){
        this.mUser = user;
    }

    public User getUser() {
        return mUser;
    }

    public void setUser(User user) {
        mUser = user;
    }
}

修改 MainActivity.java

...
//@Inject User mUser;// 注释掉这行
@Inject ClassRoom mClassRoom;// 添加这行注入
...
mUserAgeTv.setText("" + mClassRoom.getUser().getAge());
// 修改获取方式,测试注入是否成功
...

2. 生成代码对比

ClassRoom.java => ClassRoom_Factory.java

编译,查看生成代码,多出一个 ClassRoom_Factory ,效果和对 User 构造方法的注入是一样的。

重点关注 DaggerUserComponentMainActivity_MemberInjector 的变化

(1) DaggerUserComponent.java 变化不大,主要是多了一个 Provider以及对应的初始化
private Provider<ClassRoom> classRoomProvider;// 多出一个 clssRoomProvider
...
private void initialize(final Builder builder) {
  // 初始化方法变化,需要先初始化 classRoomProvider,然后再创建 mainActivityMembersInjector
    this.classRoomProvider = ClassRoom_Factory.create(User_Factory.create());
    this.mainActivityMembersInjector = MainActivity_MembersInjector.create(classRoomProvider);
}
...

注意: 此处对classRoomProvider初始化 需要调用到他所依赖的 User 所生成的User_Factory

这里也是处理依赖的地方,当 ClassRoom 生成需要 User 时,需要先提供 User 的生成者,也就是 User_Factory

(2) MainActivity_MemberInjector.java,变化也不大

因为去掉了 @Inject User mUser; 这行,因此 Provider<User> mUserProvider 也就消失了。

其余部分,主要是从 User 转换到 ClassRoom 。具体可以参考下面代码

public final class MainActivity_MembersInjector implements MembersInjector<MainActivity> {
  private final Provider<ClassRoom> mClassRoomProvider;

  public MainActivity_MembersInjector(Provider<ClassRoom> mClassRoomProvider) {
    assert mClassRoomProvider != null;
    this.mClassRoomProvider = mClassRoomProvider;
  }

  public static MembersInjector<MainActivity> create(Provider<ClassRoom> mClassRoomProvider) {
    return new MainActivity_MembersInjector(mClassRoomProvider);
  }

  @Override
  public void injectMembers(MainActivity instance) {
    if (instance == null) {
      throw new NullPointerException("Cannot inject members into a null reference");
    }
    instance.mClassRoom = mClassRoomProvider.get();
  }

  public static void injectMClassRoom(
      MainActivity instance, Provider<ClassRoom> mClassRoomProvider) {
    instance.mClassRoom = mClassRoomProvider.get();
  }
}
(3) 总结

所以综上,当需要注入的对象,依赖另一个对象时,Dagger 2 编译 生成的代码,和原本的方法区别不是很大

最大变化主要是初始化 DaggerUserComponent 时候,多了对 classRoomProvider 的初始化赋值。

A 依赖 B 时,Dagger 在生成 Component 实例的时候,会调用 BB_Factory来生成 mAProvider

然后才能对 mainActiivtyMembersInjector 进行初始化

B 配合@Module 解决多层依赖

该部分代码B

1. 总述

@Module 作用,某个模块依赖的提供者,@Provides 配合使用,其主要是下面两种情况:

  1. 需要的依赖并没有存在实例,需要 new 出来
  2. 需要的依赖来自已经创建的或者已存在的对象

2.具体使用模拟

此处先考虑实例 需要 new 出来这种情况。

添加 Subject.java

public class Subject {
    String mName;
    String mId;
    ClassRoom mClassRoom;
    User mUser;
    public Subject(ClassRoom classRoom){
        this.mName = "";
        this.mId = "";
        this.mClassRoom = classRoom;
        this.mUser = classRoom.getUser();
    }   
    ...
    ...
}

添加SubjectModule.java

@Module
public class SubjectModule {
    public SubjectModule(){

    }
    @Provides Subject provideSubject(ClassRoom classRoom){
        return new Subject(classRoom);
    }
}

修改UserComponent.java

@Component(modules = SubjectModule.class)// 修改加入 modules 依赖
...

修改 MainActivity.java

//    @Inject User mUser;
//    @Inject ClassRoom mClassRoom;
// 注释掉上面两行,添加 mSubject 的注入
    @Inject Subject mSubject;
    SubjectModule mSubjectModule;
    ...
    ...
    mSubjectModule = new SubjectModule();
// 注意此处调用方式已经变了,需要外部自己传入标记的 xxModule 类
    DaggerUserComponent.builder()
      .subjectModule(mSubjectModule)
      .build().injectTo(this);
    mUserAgeTv.setText("" + mSubject.mClassRoom.getUser().getAge());

3. 查看该部分内容对应生成的代码

加入的修改的类有些多,所以再看一次其生成代码的对应关系

(1) SubjectModule.java => SubjectModule_ProvideSubjectFactory.java

@Provides Subject provideSubject(ClassRoom classRoom) 该注解,确定了 生成的 xxx_ProvidexxxFactory 需要用到的依赖,此处是ClassRoom 类,对应的是 ClassRoom_Factory ,下面具体的代码中也可看出

public final class SubjectModule_ProvideSubjectFactory implements Factory<Subject> {
  private final SubjectModule module;

  private final Provider<ClassRoom> classRoomProvider;// @provides 标记的函数所需要的依赖

  public SubjectModule_ProvideSubjectFactory(
      SubjectModule module, Provider<ClassRoom> classRoomProvider) {
    assert module != null;
    this.module = module;
    assert classRoomProvider != null;
    this.classRoomProvider = classRoomProvider;
  }

  @Override
  public Subject get() {
    // get方法不一样,不是直接 new ,而是调用 SubjectModule.provideSubject() 方法
    // 而该方法依赖 ClassRoom 对象,因此需要调用到 classRoomProvider.get 来获取 ClassRoom 的实例
    return Preconditions.checkNotNull(
        module.provideSubject(classRoomProvider.get()),
        "Cannot return null from a non-@Nullable @Provides method");
  }
// 该部分生成方法和以前的 ClassRoom_Factory 一样,有外层依赖,需要传入
  public static Factory<Subject> create(
      SubjectModule module, Provider<ClassRoom> classRoomProvider) {
    return new SubjectModule_ProvideSubjectFactory(module, classRoomProvider);
  }
}
(2) UserComponent.java => DaggerUserComponent.java

主要关注点三个

a.该部分添加了@Component(modules = SubjectModule.class) ,整体调用上有变化,需要外部传入 SubjectModule的实例

b.同时多了一个 Provider<Subject> provideSubjectProvider

c. 因为Dagger2 通过上述 (1) 中 @Provides 所标记的函数,所需要的依赖判断出需要ClassRoom的实例提供者 ,因此有classRoomProvider

    ....  
    private Provider<Subject> provideSubjectProvider;
    ...
    ...
    @SuppressWarnings("unchecked")
    private void initialize(final Builder builder) {

      this.classRoomProvider = ClassRoom_Factory.create(User_Factory.create());

      this.provideSubjectProvider =
          SubjectModule_ProvideSubjectFactory.create(builder.subjectModule, classRoomProvider);

      this.mainActivityMembersInjector = MainActivity_MembersInjector.create(provideSubjectProvider);
    }

    @Override
    public void injectTo(MainActivity mainActivity) {
      mainActivityMembersInjector.injectMembers(mainActivity);
    }

    public static final class Builder {
      private SubjectModule subjectModule;
      ...
      public UserComponent build() {
        if (subjectModule == null) {// 2 如果为空,dagger 会自己创建一个
          this.subjectModule = new SubjectModule();
        }
        return new DaggerUserComponent(this);
      }
    // 1 构造的时候可以传入 subjectModule
      public Builder subjectModule(SubjectModule subjectModule) {
        this.subjectModule = Preconditions.checkNotNull(subjectModule);
        return this;
      }
    }
...
...

故而,配合调用链来看,上述代码注释中的 1,2需要注意

因为外部传入的 subjectModule 是由自己创建,控制的

而如果没有传入,dagger 会自动创建一个

如果当前 DaggerUserComponent 被重复使用,其 subjectModule 也会一直重复使用同一个

该部分具体看需求,不同情况不同使用

(3)MainActivity.java => MainActivity_MembersInjector.java

其中 MainActivity_MembersInjecto.java 修改很小,和上面A部分情况类似

因为去掉了 @Inject ClassRoom mClassRoom; 故而就是把A部分的中mClassRoomProvider => mSubjectModule

3. 第二种情况处理

我们需要从外面传入自己new 的实例,再调用 Dagger 来生成对应的实例。

可以借助 SubjectModule 构造函数,或者方法进行放入,如下:

SubjectModule mSubjectModule = new SubjectModule(new ClassRoom());
DaggerUserComponent.builder()
      .subjectModule(mSubjectModule)
      .build().injectTo(this);
    ...
    ClassRoom mClassRoom;
    public SubjectModule(ClassRoom classRoom){
        mClassRoom = classRoom;
    }
    ...
    @Provides Subject provideSubject(){
        return new Subject(mClassRoom);
    }
    //也可以给出对外接口
    //public void setClassRoom(ClassRoom classRoom){
    //    this.mClassRoom = classRoom;
    //}
    ...

以上就是关于 Dagger 在面对多层依赖的简单情况下,生成代码的分析

其中关于 @Module @Provides 两个注解是关键部分,一个标注当前类是Module,一个标注的函数会生成xxxModule_providexxxFactory 从而为 Component 注入提供实例。

然后附上Component Module 和被注入类的整体关系图,因为上面写的比较详细,此处就只给一个整体图了

dagger2 学习(二)

一家之言,如有错误,轻喷。

该部分代码A

该部分代码B

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

推荐阅读更多精彩内容

  • Dagger2 转载请注明原作者,如果你觉得这篇文章对你有帮助或启发,可以关注打赏。 前言本文翻译自Google ...
    轻云时解被占用了阅读 6,656评论 4 31
  • Spring Cloud为开发人员提供了快速构建分布式系统中一些常见模式的工具(例如配置管理,服务发现,断路器,智...
    卡卡罗2017阅读 134,585评论 18 139
  • Android 自定义View的各种姿势1 Activity的显示之ViewRootImpl详解 Activity...
    passiontim阅读 171,398评论 25 707
  • 前言 该部分代码 记录一下自己的基本的认识,Dagger 2 的目的是为了解决依赖问题 当我们申明一个 User ...
    搬代码白言午阅读 410评论 0 0
  • 这几天心里很烦,人生的诸多不顺总是在不经意的袭来,社会繁忙凝成了心中沉重的困惑,我自己仿佛成了一只孤独而又疲...
    玫瑰心阅读 198评论 0 1