我所了解的 Dagger2(二)

我所了解的 Dagger2(一)之后,这篇文章主要进行以下几个方面的补充:

  1. @Scope 实现一个单例;
  2. 自定义 @Scope
  3. Component 之间的拓展和依赖。

APP 内实现单例

上一篇文章中讲到了真正创建单例的方法,简单的回顾一下:

  1. 在 AppModule 中定义创建全局类实例的方法;
  1. 创建 AppComponent 来管理 AppModule;
  2. 保证 AppComponent 只有一个实例;
声明

现在我们要在项目的 Application 中创建 AppComponent,保证我们 app 全局只有一个 OkHttpClient,来演示一下单例:

@Singleton
@Component(modules = {AppModule.class})
public interface AppComponent {
    void inject(MyApplication application);
    OkHttpClient getOKHttpClient(); //非必要的,这里可以暴露方法供我们直接用 AppComponent 实例调用
}
------ 分割线 ------
@Module
public class AppModule {
    @Provides
    @Singleton
    OkHttpClient provideOkHttpClient(Interceptor interceptor) {
        return new OkHttpClient.Builder().connectTimeout(30 * 1000, TimeUnit.MILLISECONDS)
                .addInterceptor(interceptor)
                .readTimeout(30 * 1000, TimeUnit.MILLISECONDS)
                .build();
    }
    @Provides
    Interceptor provideCommonParamsInterceptor() {
        return new Interceptor() {
            @Override
            public Response intercept(Chain chain) throws IOException {
                Request request = chain.request();
                final HttpUrl newUrl = request.url().newBuilder()
                        .addQueryParameter("channel", "android")
                        .addQueryParameter("version", "1.0.0")
                        .build();
                Request newRequest = request.newBuilder().url(newUrl).build();
                return chain.proceed(newRequest);
            }
        };
    }
}
------ 分割线 ------
public class MyApplication extends Application {
    @Inject
    AppComponent mAppComponent;

    private static MyApplication sInstance;

    public static MyApplication get(){
        return sInstance;
    }

    @Override
    public void onCreate() {
        super.onCreate();
        sInstance = this;
        DaggerAppComponent.create().inject(this);
    }

    public AppComponent getAppComponent(){
        return mAppComponent;
    }
}

这里我们保证只要 Application 中只有 AppComponent,那么我们的单例就已经完成了,唯一可能感到疑惑的是 MyApplication 中的这种写法:

@Inject
AppComponent mAppComponent;

我们并没有用 @Inject 标注哪个构造函数,也没有在 Module 中提供,mAppComponent 怎么初始化?事实上,Dagger2 生成的 DaggerAppComponent 继承了 AppComponent,inject() 的时候把自身赋值给了 mAppComponent,有兴趣的同学可以看下这样写的时候 Dagger2 生成的代码,还是比较容易读懂的。

使用

那么单例已经实现了,我们看下如何在 Activity 中使用,顺便验证一下是不是单例的。现在我们需要用到的 OkHttpClient 已经包含在 AppComponent 中,当然不能再定义一个 Component 去把 AppModule 直接添加进来,所以我们需要用到 Component 的拓展和依赖。

方法一:拓展(@Subcomponent)

用于拓展原有的 Component。这里我把 @SubComponent 称作拓展,主要就是与接口继承做一下区分。

//简单的实现
@Subcomponent
public interface OkHttpSingleComponent {
    void inject(OkHttpSingleActivity secondActivity);
}
------ 分割线 ------
@Singleton
@Component(modules = {AppModule.class})
public interface AppComponent {
    void inject(MyApplication application);
    OkHttpSingleComponent plus();
    OkHttpClient getOKHttpClient();
}

实现了一个最简单的 @Subcomponent,然后在 AppComponent 中加一个方法把 OkHttpSingleComponent 返回出去,继续看一下 OkHttpSingleActivity 的实现:

public class OkHttpSingleActivity extends BaseActivity {
    @Inject
    OkHttpClient mOkHttpClient;
    @Inject
    OkHttpClient mOkHttpClient2;
    @Override
    protected void onCreate(@Nullable Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_okhttp_single);
        MyApplication.get().getAppComponent().plus().inject(this);//

        TextView tvTime = (TextView) findViewById(R.id.tv_time);
        tvTime.setText("time:" + System.currentTimeMillis());
        TextView tv = (TextView) findViewById(R.id.tv);
        tv.setText("mOkHttpClient == mOkHttpClient2 ? " + (mOkHttpClient == mOkHttpClient2) + "\n" +
                mOkHttpClient.toString());
    }
}

这里我们直接拿到 Application 中的 AppComponent,调用方法拿到刚刚定义的 OkHttpSingleComponent,然后就完成注入了,至此我们就完成了单例的 OkHttpClient。
这里为了简单没有给 @Subcomponent 添加 Module,事实上它其他的用法和 @Component 一样。当然在父 Component 中也可以添加多个 Subcomponent。下面看一下第二种实现方法。

方法二:依赖

在说 @Component 的依赖之前,要说两个关于 @Scope 的注意点

  1. 如果 Component 的 Module 中有被 @Scope 标注,则 Component 也需要用相同的 @Scope 标注;
  2. 如果 Component 依赖其他的 OtherComponent 已经被 @Scope 标注,则该 Component 也需要用 @Scope 标注,且不能与被依赖的 OtherComponent 的 @Scope 相同。

如果没有遵循以上两点,编译期间就会报错。
既然已经提到依赖需要不同的 @Scope 我们就先来自定义一个:
仿照 @Singleton 我们定义一个 @PreActivity

@Scope
@Documented
@Retention(RUNTIME)
public @interface Singleton {}
------ 分割线 ------
@Scope
@Documented
@Retention(RUNTIME)
public @interface PerActivity {}

接下来看一下依赖和使用的代码:

@PerActivity
@Component(dependencies = {AppComponent.class})
public interface OkHttpSingle2Component {
    void inject(OkHttpSingleActivity okHttpSingleActivity);
}
------ 分割线 ------
public class OkHttpSingleActivity extends BaseActivity {
    private static final String TAG = OkHttpSingleActivity.class.getSimpleName();
    @Inject
    OkHttpClient mOkHttpClient;
    @Inject
    OkHttpClient mOkHttpClient2;
    @Override
    protected void onCreate(@Nullable Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_okhttp_single);
        //MyApplication.get().getAppComponent().plus().inject(this);//方法1
        DaggerOkHttpSingle2Component.builder().appComponent(MyApplication.get().getAppComponent()).build().inject(this);//方法2
        TextView tvTime = (TextView) findViewById(R.id.tv_time);
        tvTime.setText("time:" + System.currentTimeMillis());
        TextView tv = (TextView) findViewById(R.id.tv);
        tv.setText("mOkHttpClient == mOkHttpClient2 ? " + (mOkHttpClient == mOkHttpClient2) + "\n" +
                mOkHttpClient.toString());
    }
}
  • 直接在 OkHttpSingle2Component 的 @Component 后面添加 dependencies,这里与添加 Module 类似,可以依赖多个 Component;
  • 接着添加自定义的 @Scope 用于和 AppComponent 中的 @Singleton 做区分;
  • 最后在 build 的时候需要显式的把 AppComponent 实例传进去,然后进行 inject。

我们只要保证无论在哪个 Activity 或者 Fragment 中,我们最终传进去的 AppComponent 都是 MyApplication 中的实例,那么我们就能确保最终的 OkHttpClient 为单例。

关于 Component 的组织

其实上边的例子就是 Component 的组织了,把多个页面不同的 Component 划分开来,通过 @Scope 的两个注意点 来更好的管理 Component 与 Component 以及 Component 与 Module 之间的关系。

Component.png

因为不同 @Scope 标注 Component 与 Module 组织到一起会报错,依赖关系的 Component 不能用相同的 @Scope,体现了 @Scope 的含义。
如果觉得这里没有表达清楚,强烈建议看一下 Android:dagger2 让你爱不释手-重点概念讲解、融合篇 相信会有更深的认识。

最后

如果之前有看过我所了解的 Dagger2(一)再来看这篇文章的同学,看到这里相信对 Dagger2 以及它的使用有了初步的认识。虽然公司的项目中没有用到,但是对我自身来说,阅读有使用 Dagger2 的开源项目也不再会是一头雾水。这两篇文章的举例我尽量做得循序渐进了,如果有写的不对的地方请提醒我,避免误导了后来的同学,如果觉得我整理的还不错请为我点个赞,谢谢!

已经被改的不成样的源码

参考资料

Android:dagger2 让你爱不释手-重点概念讲解、融合篇
Android 常用开源工具(2)-Dagger2 进阶
详解 Dagger2

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

推荐阅读更多精彩内容