最近项目用到mvp+dagger,由于使用过程中始终对dagger比较模糊,所以对git上开源GeekNews的dagger断点分析。git上源码地址
首先我们先来了解一下dagger的基本知识:
Dagger 主要分三块:
@Inject:需要注入依赖的地方,Dagger 会构造一个该类的实例并满足它所需要的依赖;
@Module:依赖的提供者,Module 类中的方法专门提供依赖,并用 @Provides 注解标记;
@Component:依赖的注入者,是 @Inject 和 @Module 的桥梁,它从 @Module 中获取依赖并注入给 @Inject。
对于以上关系,一句话解释就是:模块(Module)负责提供依赖,组件(Component)负责注入依赖。
Dagger 的其他注解:
1、@Scope: Dagger 可以通过自定义注解限定注解作用域,参考前面的 @ActivityScope。
2、@Qualifier:限定符,当类的类型不足以鉴别一个依赖的时候,我们就可以使用这个注解来区分。例如:在 Android 中,我们会需要不同类型的 Context,所以我们可以定义 @Qualifier 注解 @ForApplication
和@ForActivity
,这样当注入一个 Context 的时候,我们就可以告诉 Dagger 我们想要哪种类型的 Context。
3、@Singleton:单例模式,依赖的对象只会被初始化一次
GeekNews 的dagger结构:
ActivityScope、ContextLife、FragmentScope分别设置了注入activity、application、fragment的生命周期范围。
我们开始断点调试:
WelcomeActivity继承BaseActivity,首先先走BaseActivity的onCreate()
然后执行到注入方法:
紧接着执行子类的WelcomeActivity的initInject方法:
然后调用BaseAcitivity的getActivityComponent方法:
以上的三张图可以看出将application实例传入DaggerAppComponent的appModule。那么DaggerAppComponent是什么呢?从代码里看到
public final class DaggerAppComponent implements AppComponent 那就是说DaggerAppComponent也是一个容器,装Application的容器。
以上四张截图可以看出我们在AppComponent容器里定义的三个对象provideApplicationContextProvider、provideRetrofitHelperProvider、provideRealmHelperProvider已经被赋值了。
AppComponent的容器定义如下:
package com.codeest.geeknews.di.component;
import com.codeest.geeknews.app.App;
import com.codeest.geeknews.di.ContextLife;
import com.codeest.geeknews.di.module.AppModule;
import com.codeest.geeknews.model.db.RealmHelper;
import com.codeest.geeknews.model.http.RetrofitHelper;
import javax.inject.Singleton;
import dagger.Component;
/**
* Created by codeest on 16/8/7.
*/
@Singleton
@Component(modules = AppModule.class)
public interface AppComponent {
@ContextLife("Application")
App getContext(); // 提供App的Context
RetrofitHelper retrofitHelper(); //提供http的帮助类
RealmHelper realmHelper(); //提供数据库帮助类
}
以上两张图将获取welcomeActivity传入到ActivityModule里边,为Activity容器的注入做铺垫。
图18是图16的DaggerActivityComponent.build()方法实现。
以上两张图看到DaggerActivityComponent.build()方法里welcomePresenterProvider已经被赋值了。
到此执行完WelcomeActivity的DaggerActivityComponent(),紧接着调用inject()方法:
public final class DaggerActivityComponent implements ActivityComponent,DaggerActivityComponent实现ActivityComponent,所以也是一个容器。
ActivityComponent的代码如下:
package com.codeest.geeknews.di.component;
import android.app.Activity;
import com.codeest.geeknews.di.ActivityScope;
import com.codeest.geeknews.di.module.ActivityModule;
import com.codeest.geeknews.ui.main.activity.MainActivity;
import com.codeest.geeknews.ui.main.activity.WelcomeActivity;
import com.codeest.geeknews.ui.zhihu.activity.SectionActivity;
import com.codeest.geeknews.ui.zhihu.activity.ThemeActivity;
import com.codeest.geeknews.ui.zhihu.activity.ZhihuDetailActivity;
import dagger.Component;
@ActivityScope
@Component(dependencies = AppComponent.class, modules = ActivityModule.class)
public interface ActivityComponent {
Activity getActivity();
void inject(WelcomeActivity welcomeActivity);
}
所以DaggerActivityComponent 实现了ActivityComponent的inject()
图22是activity获取presenter,以下4张图是获取的过程:
可以图27可以看出presenter已经不为空了。图24和图26,可以看出网络实例mRetrofitHelper 已经注入到presenter种。
最后可以看出RxPresenter也和View关联起来了。
由于本人水平有限,有什么错误的地方或者写的不够好,欢迎指出!大家一起共同学习哈!!!