[Android 学习笔记] Dagger2 依赖注入由浅入深 (3)

Scope 是 Dagger2 库中比较难理解一个概念, 它可以翻译为"作用域", 进一步解释就是"创建出的对象的生命周期".
Dagger2 中 Module 只负责创建所需的对象, Module 中创建的对象由 Component 负责缓存和复用, 至于这个对象是每次调用都要新创建, 还是全局都复用同一个对象, 这些是在自动生成的实现了 Component 接口的代码中.

没有使用 Scope 注解的场景

定义的 SimpleActivityComponent 接口:

@Component(modules = { UserServerModule.class })
public interface SimpleActivityComponent {
    UserServer getUserServer();
}

自动生成的 DaggerSimpleActivityComponent 类:

public final class DaggerSimpleActivityComponent implements SimpleActivityComponent {
  private DaggerSimpleActivityComponent(Builder builder) {}

  public static Builder builder() {
    return new Builder();
  }

  public static SimpleActivityComponent create() {
    return new Builder().build();
  }

  @Override
  public UserServer getUserServer() {
    return UserServerModule_ProvideUserServerFactory.proxyProvideUserServer();
// 该方法每次都会去  UserServerModule.provideUserServer() 方法, 所以每次都会去调用 new UserServer("kimi"), 这样每次调用 getUserServer() 方法都是去新创建一个 UserServer 对象
  }

  public static final class Builder {
    private Builder() {}

    public SimpleActivityComponent build() {
      return new DaggerSimpleActivityComponent(this);
    }

    /**
     * @deprecated This module is declared, but an instance is not used in the component. This
     *     method is a no-op. For more, see https://google.github.io/dagger/unused-modules.
     */
    @Deprecated
    public Builder userServerModule(UserServerModule userServerModule) {
      Preconditions.checkNotNull(userServerModule);
      return this;
    }
  }
}

上面的代码中可以看出, 每次调用 getUserServer() 方法都是去新创建一个 UserServer 对象.

引入 Scope 注解解决对象的复用问题

javax.inject 库中自带一个 Singleton 注解, 它的类定义上添加了 @Scope 注解.

package javax.inject;

import java.lang.annotation.Documented;
import java.lang.annotation.Retention;
import static java.lang.annotation.RetentionPolicy.RUNTIME;

/**
 * Identifies a type that the injector only instantiates once. Not inherited.
 *
 * @see javax.inject.Scope @Scope
 */
@Scope
@Documented
@Retention(RUNTIME)
public @interface Singleton {}

在自定义 Module 的 provide 方法上添加 @Singleton 注解

import dagger.Module;
import dagger.Provides;
import javax.inject.Singleton;

@Module
public class UserServerModule {

    @Singleton  // 添加 Singleton 注解
    @Provides
    static UserServer provideUserServer() {
        return new UserServer("kimi");
    }
}

然后再定义的 SimpleActivityComponent 接口类定义上添加 @Singleton 注解:

import dagger.Component;
import javax.inject.Singleton;

@Singleton  // 添加 Singleton 注解
@Component(modules = { UserServerModule.class })
public interface SimpleActivityComponent {
    UserServer getUserServer();
}

编译后再看自动生成的 DaggerSimpleActivityComponent 类:

import dagger.internal.DoubleCheck;
import dagger.internal.Preconditions;
import javax.inject.Provider;

public final class DaggerSimpleActivityComponent implements SimpleActivityComponent {

  // 添加了一个 provideUserServerProvider 变量
  private Provider<UserServer> provideUserServerProvider;

  private DaggerSimpleActivityComponent(Builder builder) {
    initialize(builder);
  }

  public static Builder builder() {
    return new Builder();
  }

  public static SimpleActivityComponent create() {
    return new Builder().build();
  }

// 多了一个 initialize 方法用于给变量 provideUserServerProvider 赋值
// 这里介绍一下 DoubleCheck 类, 它会缓存传值, 所以每次调用它的 get() 方法时, 返回的都是同一个对象 
  @SuppressWarnings("unchecked")
  private void initialize(final Builder builder) {
    this.provideUserServerProvider =
    DoubleCheck.provider(UserServerModule_ProvideUserServerFactory.create());
  }

  @Override
  public UserServer getUserServer() {
    return provideUserServerProvider.get();  // 所以每次再调用  get() 方法都是返回同一个对象, 不是每次再创建新对象
  }

  public static final class Builder {
    private Builder() {}

    public SimpleActivityComponent build() {
      return new DaggerSimpleActivityComponent(this);
    }

    /**
     * @deprecated This module is declared, but an instance is not used in the component. This
     *     method is a no-op. For more, see https://google.github.io/dagger/unused-modules.
     */
    @Deprecated
    public Builder userServerModule(UserServerModule userServerModule) {
      Preconditions.checkNotNull(userServerModule);
      return this;
    }
  }
}

从上面的代码和添加的注释可以看出:
添加 Scope 注解使 Component 对象强引用了 Module 中提供的对象, 当 Component 对象存在时, Module 提供的对象也是存在的, 不会被回收, 所以它们两个是同样的生命周期.
不用 Scope 注解时, Component 对象每次都会创建新的 Module 提供的对象, Component 和 Module 提供的对象就不是同一个生命周期.

@Singleton

Singleton 给我们的第一印象是它注解生成的对象应该是全局唯一的, 单例的, 但是通过 demo 中可以看出 @Singleton 注解的对象也是和 Component 相同生命周期的, 如果 Component 被回收了那么 @Singleton 注解的对象也会被回收, 所以它不是全局唯一, 如果 Component 被创建了多次, 那么内存中也会出现多个 @Singleton 注解的对象, 不能被它的命名迷惑了.

自定义 Scope 注解

javax.inject 包中只提供了一个 Scope 注解, 也就是 @Singleton, 但是开发者也可以自定义 Scope 注解.

import java.lang.annotation.Retention;
import java.lang.annotation.RetentionPolicy;
import javax.inject.Scope;

@Scope // 注意
@Retention(RetentionPolicy.CLASS)
public @interface CustomScope {
}

自定义 Scope 注解很简单, 但是需要用 @Scope 来注解自定义的注解类, 这样做的目的是方便 AnnotationProcessor 在编译时能找到哪些是自定义的 Scope 注解.

自定义注解用于区分各个 Module 的生命周期, 就拿日常 Android 开发来说, 有的对象在整个 APP 生命周期内都被引用, 所以只有完全退出 APP 后, 这个对象才会被回收, 而有的对象只有在某个 Activity 生命周期内被引用了, 当退出这个 Activity 时, 这个对象也会被回收.
所以我们可以定义两个 Scope 注解:

@Scope
@Retention(RetentionPolicy.CLASS)
public @interface ApplicationScope {
}


@Scope
@Retention(RetentionPolicy.CLASS)
public @interface LoginActivityScope {
}

ApplicationScope 注解用于注解那些与 APP 整个生命周期绑定的对象, LoginActivityScope 注解可以用于注解只在 LoginActivity 的生命周期内被引用的对象, 比如 LoginServer 对象.

参考资料

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

推荐阅读更多精彩内容