Dagger2在Android开发中的新用法.

本文假设读者已经有必定Dagger2使用经验html

使用疑惑

以前工做中一直在使用dagger2进行开发,用起来确实很爽,可是我从我第一次使用我就一直有一个问题或者说疑问(本人才疏学浅脑子不够使),一般状况下咱们有以下清单android

MyApplicationMyAppComponentMyAppModule
ActActivity,ActComponent,ActModulegit

简单解释下,MyAppModule提供全局单例功能,好比打印日志,ActModule提供Activity级别的功能好比发起网络请求(只是举个栗子),如今咱们但愿在发起网络请求的时候打印日志,那么解决方法也很简单——SubComponent或者Component(dependencies=X.class)github

因而咱们首先在MyApplication中初始化MyAppcomponent(使用抽象类实现单例)网络

@Component(modules = MyAppModule.class)
public abstract class MyAppComponent {
    ......
   //使用SubComponent功能来完成component的组合
   abstract ActComponent plus();
}
@Subcomponent(modules = ActModule.class)
public interface ActComponent {
    void inject(ActActivity act);
}
public class MyApplication extends Application {
    @Override
    public void onCreate() {
        super.onCreate();
        MyAppComponent.getInstance().inject(this);
    }
}

而后就是就在Activity中使用ActComponent来提供注入功能,代码看上去就像以下...ide

MyAppComponent.getInstance()
                .plus()
                .inject(this);

为神马我使用的明明是ActComponent,关MyAppComponent什么事?(我最开始学习使用dagger2的时候彻底没法接受这种写法),并且这彷佛不太符合依赖注入的一个根本原则a class shouldn’t know anything about how it is injected.学习

新用法

谷歌爸爸很明显也注意到了这个问题,谁叫Dagger2Android开发中也那么火呢,因而在Dagger2新版本中咱们有了一个新东西dagger.androidui

Gradle引入方式this

//dagger2
    compile 'com.google.dagger:dagger:2.11'
    compile 'com.google.dagger:dagger-android:2.11'
    compile 'com.google.dagger:dagger-android-support:2.11'
    annotationProcessor 'com.google.dagger:dagger-compiler:2.11'
    annotationProcessor 'com.google.dagger:dagger-android-processor:2.11'

Demo地址在 https://github.com/hanliuxin5...google

Talk is cheap, show me the code

结合Demo和官方文档粗略翻译以下

  1. AppComponent中安装AndroidInjectionModule

@Component(modules = {AndroidInjectionModule.class})
public interface AppComponent {
    //....
}

2.编写实现了AndroidInjector<YourActivity>Lychee3Activity

@Subcomponent(modules = ...)
public interface ActSubComponent extends AndroidInjector<Lychee3Activity> {
    @Subcomponent.Builder
    public abstract class Builder extends AndroidInjector.Builder<Lychee3Activity> {
    }
}

3.定义了ActSubComponent后,将其安装在绑定了ActSubComponent.BuilderModule中,而且将该Module安装在咱们的AppComponent

@Module(subcomponents = {ActSubComponent.class})
public abstract class BuildersModule {
    @Binds
    @IntoMap
    @ActivityKey(Lychee3Activity.class)
    abstract AndroidInjector.Factory<? extends Activity> lychee3Activity(ActSubComponent.Builder builder);
    }
@Component(modules = {AndroidInjectionModule.class,
        BuildersModule.class})
public interface AppComponent {
    //....
}

可是若是你的ActSubComponent若同咱们在步骤2中定义的同样,无论在类中仍是在其Builder中没有的方法和超类型,你能够用下面的代码跳过2,3步骤

原文 Pro-tip: If your subcomponent and its builder have no other methods or supertypes than the ones mentioned in step #2, you can use @ContributesAndroidInjector to generate them for you

@ContributesAndroidInjector
    abstract Lychee2Activity lychee2Activity();

4.让你的MyApplication实现HasActivityInjector,而且注入DispatchingAndroidInjector

public class MyApplication extends Application implements HasActivityInjector {
    @Inject
    DispatchingAndroidInjector<Activity> dispatchingAndroidInjector;

    @Override
    public void onCreate() {
        super.onCreate();
                DaggerAppComponent.builder().AppContent(this).build().inject(this);//最好结合demo来看,否则AppContent是啥你不知道
    }

    @Override
    public AndroidInjector<Activity> activityInjector() {
        return dispatchingAndroidInjector;
    }
}

5.最后,在你Lychee3ActivityLychee2Activity中的onCreate中,调super.onCreate()以前调用AndroidInjection.inject(this);

public class Lychee2Activity extends AppCompatActivity {
  public void onCreate(Bundle savedInstanceState) {
    AndroidInjection.inject(this);
    super.onCreate(savedInstanceState);
  }
}

至此,新东西的使用差很少就到这了,可是为何我会有一种“天,怎么愈来愈复杂啦”的感受呢...

第一次写文章,有什么不足的地方或者你以为很不爽的东西欢迎交流指出。

参考文章
https://google.github.io/dagg...
https://android.jlelse.eu/and...

相关文章
相关标签/搜索