官方文档链接: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
和@Subcomponent.Builder
注解的内部类继承AndroidInjector.Builder
:@Subcomponent(modules = ...) public interface YourActivitySubcomponent extends AndroidInjector
{ @Subcomponent.Builder public abstract class Builder extends AndroidInjector.Builder {} } -
定义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
对象去让activityInjector()
方法返回:public class YourApplication extends Application implements HasActivityInjector { @Inject DispatchingAndroidInjector
dispatchingActivityInjector; @Override public void onCreate() { super.onCreate(); DaggerYourApplicationComponent.create() .inject(this); } @Override public AndroidInjector 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传入inject(Activity)
方法中。DispatchingAndroidInjector
对象通过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 fragmentInjector;
@Override
public void onCreate(Bundle savedInstanceState) {
AndroidInjection.inject(this);
super.onCreate(savedInstanceState);
// ...
}
@Override
public AndroidInjector 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 {
@Subcomponent.Builder
public abstract class Builder extends AndroidInjector.Builder {}
}
@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在安卓中使用的关键点就是不让被注入的类知道注入的细节。部分地方由于不理解,只能照着语义翻译,若有人发现翻译错误的,还望在评论中指出,谢了!