翻译Dagger 2在安卓中使用

官方文档链接:https://google.github.io/dagger/android.html

1.前言


Dagger 2相比其它依赖注入框架的主要优势之一,严格的代码生成(没有反射),意味着可以在安卓应用中使用。但是,在安卓中使用,仍然有些事项需注意。

2.设计思想


虽然安卓是用Java编写的,但两者就风格而言完全不一样,因为得考虑移动平台独特的表现方式。许多常见的用于Java代码的模式无法用于安卓代码,甚至许多《Effective Java》书中的建议在安卓看来都是不恰当的。

为了达到既符合语言习惯又轻量的代码目标,Dagger依靠混淆来接着处理编译后的字节码。Dagger通过使用不同的工具链产生在两种环境下都有效执行的字节码,使代码在服务器和安卓上不管是看还是感觉都很自然。此外,Dagger有个明确的目标,确保生成的Java代码总能支持混淆优化(假设在安卓上使用混淆)。

当然,不是所有问题都能用那样的方式解决,但这是能提供安卓特有兼容的主要机制。

3.基本使用


使用Dagger编写安卓应用的主要困难之一,许多安卓框架类由系统自身初始化,像Activity和Fragment。但只有在Dagger创建所有的注入对象时,它才能完美工作。这导致需要在生命周期方法中完成对象注入,将有许多类像下面这样:

public class FrombulationActivity extends Activity {
  @Inject Frombulator frombulator;

  @Override
  public void onCreate(Bundle savedInstanceState) {
    super.onCreate(savedInstanceState);
    // DO THIS FIRST. Otherwise frombulator might be null!
    ((SomeApplicationBaseType) getContext().getApplicationContext())
        .getApplicationComponent()
        .newActivityComponentBuilder()
        .activity(this)
        .build()
        .inject(this);
    // ... now you can write the exciting code
  }
}

这有几个问题(dagger.android包中的类提供了简化这种模式的方法):

  • 样板代码使以后重构变得困难。随着越来越多的开发者复制粘贴这段代码,越来越少的人知道它实际作用。
  • 更严重的是,它需要请求注入的对象(FrombulationActivity)知道注入器。即使这是通过接口代替实体类,但仍打破了依赖注入的核心原则:一个类不应该知道任何关于如何被注入的。

4.给Activity注入对象


  • 为Application级的Component添加AndroidInjectionModule来确保Dagger必要的所有基础类型依赖是有效的。

  • 从创建@Subcomponent注解的接口继承AndroidInjector<YourActivity>@Subcomponent.Builder注解的内部类继承AndroidInjector.Builder<YourActivity>

    @Subcomponent(modules = ...)
    public interface YourActivitySubcomponent extends AndroidInjector<YourActivity> {
      @Subcomponent.Builder
      public abstract class Builder extends AndroidInjector.Builder<YourActivity> {}
    }
    
  • 定义Subcomponent之后,通过定义个Module依赖Subcomponent及其Builder,并将它添加到Application级的Component中,来添加Subcomponent到Component层次结构中:

    @Module(subcomponents = YourActivitySubcomponent.class)
    abstract class YourActivityModule {
      @Binds
      @IntoMap
      @ActivityKey(YourActivity.class)
      abstract AndroidInjector.Factory<? extends Activity>
          bindYourActivityInjectorFactory(YourActivitySubcomponent.Builder builder);
    }
    
    @Component(modules = {..., YourActivityModule.class})
    interface YourApplicationComponent {}
    

    如果Subcomponent及其Builder没有其它的方法,可以使用@ContributesAndroidInject注解帮助生成样板代码。在Module中添加抽象方法返回需要的Activity,并用@ContributesAndroidInject注解,且指定Subcomponent依赖的模块,来取代上述的2、3步。如果Subcomponent需要作用域,可以给方法使用作用域注解。

    @ActivityScope
    @ContributesAndroidInjector(modules = { /* modules to install into the subcomponent */ })
    abstract YourActivity contributeYourActivityInjector();
    
  • 接着,使Application实现HasActivityInjector,且用@Inject注解一个DispatchingAndroidInjector<Activity>对象去让activityInjector()方法返回:

    public class YourApplication extends Application implements HasActivityInjector {
      @Inject DispatchingAndroidInjector<Activity> dispatchingActivityInjector;
    
      @Override
      public void onCreate() {
        super.onCreate();
        DaggerYourApplicationComponent.create()
            .inject(this);
      }
    
      @Override
      public AndroidInjector<Activity> activityInjector() {
        return dispatchingActivityInjector;
      }
    }
    
  • 最后,在需要的Activity的onCreate()方法中,于super.onCreate()方法之前调用AndroidInjection.inject(this)方法:

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

工作原理?(此处可以参考另一篇博客,有更详细的分析)

通过AndroidInjection.inject()方法从Application中获取DispatchingAndroidInjector<Activity>对象,同时将Activity传入inject(Activity)方法中。DispatchingAndroidInjector<Activity>对象通过Activity类查询AndroidInjector.Factory(就是YourActivitySubcomponent.Builder),创建ActivityInjector(就是YourActivitySubcomponent),并将Activity传入它的inject(Activity)方法中。

5.给Fragment注入对象


给Fragment注入和给Activity注入一样简单。以同样的方式定义Subcomponent,将参数中的Activity替换为Fragment,把@ActivityKey替换为@FragmentKey,以及HasActivityInjector替换为HasFragmentInjector。注入过程发生的地方,也由Activity的onCreate()方法替换为Fragment的onAttach()方法。

不像为Activity绑定Module,可以选择在哪为Fragment绑定Module。可以让某Fragment级的Component作为另一个Fragment级(也可以是Activity级或者Application级)Component的Subcomponent,这都取决于Fragment需要的其它依赖关系。当决定Fragment级Component的位置之后,使相关Fragment实现HasFragmentInjector。举个例子,假设Fragment需要成为YourActivitySubcomponent的依赖,代码将如下所示:

public class YourActivity extends Activity
    implements HasFragmentInjector {
  @Inject DispatchingAndroidInjector<Fragment> fragmentInjector;

  @Override
  public void onCreate(Bundle savedInstanceState) {
    AndroidInjection.inject(this);
    super.onCreate(savedInstanceState);
    // ...
  }

  @Override
  public AndroidInjector<Fragment> fragmentInjector() {
    return fragmentInjector;
  }
}

public class YourFragment extends Fragment {
  @Inject SomeDependency someDep;

  @Override
  public void onAttach(Activity activity) {
    AndroidInjection.inject(this);
    super.onAttach(activity);
    // ...
  }
}

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

@Module(subcomponents = YourFragmentSubcomponent.class)
abstract class YourFragmentModule {
  @Binds
  @IntoMap
  @FragmentKey(YourFragment.class)
  abstract AndroidInjector.Factory<? extends Fragment>
      bindYourFragmentInjectorFactory(YourFragmentSubcomponent.Builder builder);
}

@Subcomponent(modules = { YourFragmentModule.class, ... }
public interface YourActivityOrYourApplicationComponent { ... }

6.基本类型


因为DispatchingAndroidInjector在运行时通过类查询相关的AndroidInjector.Factory,可以定义基本类型实现HasActivityInjector/HasFragmentInjector/etc,同时调用AndroidInjection.inject()方法,那它们的子类只需要添加相应的@Subcomponent注解的依赖。如果没有复杂的类层级结构,可使用Dagger提供的基类,例如DaggerActivity和DaggerFragment。Dagger还提供了DaggerApplication,只要继承它且重写applicationInjector()方法,返回应该注入Application中Component。基本类型还有下面这些:

  • DaggerService和DaggerIntentService
  • DaggerBroadcastReceiver
  • DaggerContentProvider

只有当BroadcastReceiver在清单文件中被注册,DaggerBroadcastReceiver才可以使用。若BroadcastReceiver在代码中注册,建议使用构造函数注入。

7.支持的库


对于使用安卓支持库的用户,有相似的类型存在于dagger.android.support包中。若使用支持库中的Fragment,依赖关系为AndroidInjector.Factory<? extends android.support.v4.app.Fragment>;若使用支持库中的AppCompatActivity,应该继续实现AndroidInjector.Factory<? extends Activity>而不是<? extends AppCompatActivity>(或者FragmentActivity)。

8.Dagger的安卓库


build.gradle中添加以下内容:

dependencies {
  compile 'com.google.dagger:dagger-android:2.x'
  compile 'com.google.dagger:dagger-android-support:2.x' // if you use the support libraries
  annotationProcessor 'com.google.dagger:dagger-android-processor:2.x'
}

9.什么时候注入


尽可能优先使用构造函数注入,因为javac将确保属性在被设置前没有被引用,以避免空指针异常,所以越早注入越好。为了这个原因,DaggerActivity在onCreate()方法中,于super.onCreate()方法前调用AndroidInjection.inject();DaggerFragment在onAttach()方法中同样如此,还能避免Fragment重连后对象不一致。

对于Activity而言,在super.onCreate()方法之前调用AndroidInjection.inject()方法是很重要的,因为当配置更改时,会调用super去连接之前Activity关联的Fragment,并依次注入。为了保证Fragment成功注入,Activity必须已经注入。若使用ErrorProne工具,可在编译时发现调用顺序上的错误。

10.AndroidInjector.Factory作用域


AndroidInjector.Factory有意成为无状态(无属性)的接口,那样就不需要管理被注入对象状态。DispatchingAndroidInjector通过Provider(注解的方法)获取AndroidInjector.Factory对象,那样就不需要明确持有Factory实例。因为AndroidInjector.Builder的实现由Dagger生成,持有待被注入的Activity/Fragment/etc实例。若给提供注入对象的方法加上作用域,将会导致编译时错误。如果也认同AndroidInjector.Factory不该持有被注入对象实例,需要使用@SuppressWarnings("dagger.android.ScopedInjectoryFactory")注解Module中的方法来忽略这个问题。

11.总结


Dagger 2在安卓中使用的关键点就是不让被注入的类知道注入的细节。部分地方由于不理解,只能照着语义翻译,若有人发现翻译错误的,还望在评论中指出,谢了!

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