博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
用 Dagger 2 实现依赖注入
阅读量:6458 次
发布时间:2019-06-23

本文共 15783 字,大约阅读时间需要 52 分钟。

本文讲的是用 Dagger 2 实现依赖注入,
  • 原文地址:
  • 原文作者:
  • 译文出自:
  • 译者: 
  • 校对者:, 

用 Dagger 2 实现依赖注入

概要

很多 Android 应用依赖于一些含有其它依赖的对象。例如,一个 Twitter API 客户端可能需要通过  之类的网络库来构建。要使用这个库,你可能还需要添加  这样的解析库。另外,实现认证或缓存的库可能需要使用  或其它通用存储方式。这就需要先把它们实例化,并创建一个隐含的依赖链。

如果你不熟悉依赖注入,看看短视频。

Dagger 2 为你解析这些依赖,并生成把它们绑定在一起的代码。也有很多其它的 Java 依赖注入框架,但它们中大多数是有缺陷的,比如依赖 XML,需要在运行时验证依赖,或者在起始时造成性能负担。  纯粹依赖于 Java 以及编译时检查来分析并验证依赖。它被认为是目前最高效的依赖注入框架之一。

优点

这是使用 Dagger 2 的一系列其它优势:

  • 简化共享实例访问。就像  库简化了引用View, event handler 和 resources 的方式一样,Dagger 2 提供了一个简单的方式获取对共享对象的引用。例如,一旦我们在 Dagger 中声明了 MyTwitterApiClient 或 SharedPreferences 的单例,就可以用一个简单的 @Inject 标注来声明域:
public class MainActivity extends Activity {   @Inject MyTwitterApiClient mTwitterApiClient;   @Inject SharedPreferences sharedPreferences;   public void onCreate(Bundle savedInstance) {       // assign singleton instances to fields       InjectorClass.inject(this);   }
  • 容易配置复杂的依赖关系。 对象创建是有隐含顺序的。Dagger 2 遍历依赖关系图,并且。而且,它可以节约大量的样板代码,使你不再需要手写,手动获取引用并把它们传递给其他对象作为依赖。它也简化了重构,因为你可以聚焦于构建模块本身,而不是它们被创建的顺序。

  • 更简单的单元和集成测试 因为依赖图是为我们创建的,我们可以轻易换出用于创建网络响应的模块,并模拟这种行为。

  • 实例范围 你不仅可以轻易地管理持续整个应用生命周期的实例,也可以利用 Dagger 2 来定义生命周期更短(比如和一个用户 session 或 Activity 生命周期相绑定)的实例。

设置

默认的 Android Studio 不把生成的 Dagger 2 代码视作合法的类,因为它们通常并不被加入 source 路径。但引入 android-apt 插件后,它会把这些文件加入 IDE classpath,从而提供更好的可见性。

确保 到最新的 Gradle 版本以使用最新的 annotationProcessor 语法:

dependencies {    // apt command comes from the android-apt plugin    compile "com.google.dagger:dagger:2.9"    annotationProcessor "com.google.dagger:dagger-compiler:2.9"    provided 'javax.annotation:jsr250-api:1.0'}

注意 provided 关键词是指只在编译时需要的依赖。Dagger 编译器生成了用于生成依赖图的类,而这个依赖图是在你的源代码中定义的。这些类在编译过程中被添加到你的IDE classpath。annotationProcessor 关键字可以被 Android Gradle 插件理解。它不把这些类添加到 classpath 中,而只是把它们用于处理注解。这可以避免不小心引用它们。

创建单例

dagger_general.pngDagger 注入概要

最简单的例子是用 Dagger 2 集中管理所有的单例。假设你不用任何依赖注入框架,在你的 Twitter 客户端中写下类似这些的东西:

OkHttpClient client = new OkHttpClient();// Enable caching for OkHttpint cacheSize = 10 * 1024 * 1024; // 10 MiBCache cache = new Cache(getApplication().getCacheDir(), cacheSize);client.setCache(cache);// Used for caching authentication tokensSharedPreferences sharedPrefeences = PreferenceManager.getDefaultSharedPreferences(this);// Instantiate GsonGson gson = new GsonBuilder().create();GsonConverterFactory converterFactory = GsonConverterFactory.create(gson);// Build RetrofitRetrofit retrofit = new Retrofit.Builder()                                .baseUrl("https://api.github.com")                                .addConverterFactory(converterFactory)                                .client(client)  // custom client                                .build();

声明你的单例

你需要通过创建 Dagger 2 模块定义哪些对象应该作为依赖链的一部分。例如,假设我们想要创建一个 Retrofit 单例,使它绑定到应用生命周期,对所有的 Activity 和 Fragment 都可用,我们首先需要使 Dagger 意识到他可以提供 Retrofit 的实例。

因为需要设置缓存,我们需要一个 Application context。我们的第一个 Dagger 模块,AppModule.java,被用于提供这个依赖。我们将定义一个 @Provides 注解,标注带有Application 的构造方法:

@Modulepublic class AppModule {    Application mApplication;    public AppModule(Application application) {        mApplication = application;    }    @Provides    @Singleton    Application providesApplication() {        return mApplication;    }}

我们创建了一个名为 NetModule.java 的类,并用 @Module 来通知 Dagger,在这里查找提供实例的方法。

返回实例的方法也应当用 @Provides 标注。Singleton 标注通知 Dagger 编译器,实例在应用中只应被创建一次。在下面的例子中,我们把 SharedPreferencesGsonCache,OkHttpClient, 和 Retrofit 设置为在依赖列表中可用的类型。

@Modulepublic class NetModule {    String mBaseUrl;    // Constructor needs one parameter to instantiate.      public NetModule(String baseUrl) {        this.mBaseUrl = baseUrl;    }    // Dagger will only look for methods annotated with @Provides    @Provides    @Singleton    // Application reference must come from AppModule.class    SharedPreferences providesSharedPreferences(Application application) {        return PreferenceManager.getDefaultSharedPreferences(application);    }    @Provides    @Singleton    Cache provideOkHttpCache(Application application) {         int cacheSize = 10 * 1024 * 1024; // 10 MiB        Cache cache = new Cache(application.getCacheDir(), cacheSize);        return cache;    }   @Provides    @Singleton   Gson provideGson() {         GsonBuilder gsonBuilder = new GsonBuilder();       gsonBuilder.setFieldNamingPolicy(FieldNamingPolicy.LOWER_CASE_WITH_UNDERSCORES);       return gsonBuilder.create();   }   @Provides   @Singleton   OkHttpClient provideOkHttpClient(Cache cache) {      OkHttpClient client = new OkHttpClient();      client.setCache(cache);      return client;   }   @Provides   @Singleton   Retrofit provideRetrofit(Gson gson, OkHttpClient okHttpClient) {      Retrofit retrofit = new Retrofit.Builder()                .addConverterFactory(GsonConverterFactory.create(gson))                .baseUrl(mBaseUrl)                .client(okHttpClient)                .build();        return retrofit;    }}

注意,方法名称(比如 provideGson()provideRetrofit() 等)是没关系的,可以任意设置。@Provides 被用于把这个实例化和其它同类的模块联系起来。@Singleton 标注用于通知 Dagger,它在整个应用的生命周期中只被初始化一次。

一个 Retrofit 实例依赖于一个 Gson 和一个 OkHttpClient 实例,所以我们可以在同一个类中定义两个方法,来提供这两种实例。@Provides 标注和方法中的这两个参数将使 Dagger 意识到,构建一个 Retrofit 实例 需要依赖 Gson 和 OkHttpClient

定义注入目标

Dagger 使你的 activity, fragment, 或 service 中的域可以通过 @Inject 注解和调用 inject()方法被赋值。调用 inject() 将会使得 Dagger 2 在依赖图中寻找合适类型的单例。如果找到了一个,它就把引用赋值给对应的域。例如,在下面的例子中,它会尝试找到一个返回MyTwitterApiClient 和SharedPreferences 类型的 provider:

public class MainActivity extends Activity {   @Inject MyTwitterApiClient mTwitterApiClient;   @Inject SharedPreferences sharedPreferences;  public void onCreate(Bundle savedInstance) {       // assign singleton instances to fields       InjectorClass.inject(this);   }

Dagger 2 中使用的注入者类被称为 component。它把先前定义的单例的引用传给 activity, service 或 fragment。我们需要用 @Component 来注解这个类。注意,需要被注入的 activity, service 或 fragment 需要在这里使用 inject() 方法注入:

@Singleton@Component(modules={AppModule.class, NetModule.class})public interface NetComponent {   void inject(MainActivity activity);   // void inject(MyFragment fragment);   // void inject(MyService service);}

注意 基类不能被作为注入的目标。Dagger 2 依赖于强类型的类,所以你必须指定哪些类会被定义。(有一些 帮助你绕开这个问题,但这样做的话,代码可能会变得更复杂,更难以追踪。)

生成代码

Dagger 2 的一个重要特点是它会为标注 @Component 的接口生成类的代码。你可以使用带有Dagger (比如 DaggerTwitterApiComponent.java) 前缀的类来为依赖图提供实例,并用它来完成用 @Inject 注解的域的注入。 参见。

实例化组件

我们应该在一个 Application 类中完成这些工作,因为这些实例应当在 application 的整个周期中只被声明一次:

public class MyApp extends Application {    private NetComponent mNetComponent;    @Override    public void onCreate() {        super.onCreate();        // Dagger%COMPONENT_NAME%        mNetComponent = DaggerNetComponent.builder()                // list of modules that are part of this component need to be created here too                .appModule(new AppModule(this)) // This also corresponds to the name of your module: %component_name%Module                .netModule(new NetModule("https://api.github.com"))                .build();        // If a Dagger 2 component does not have any constructor arguments for any of its modules,        // then we can use .create() as a shortcut instead:        //  mNetComponent = com.codepath.dagger.components.DaggerNetComponent.create();    }    public NetComponent getNetComponent() {       return mNetComponent;    }}

如果你不能引用 Dagger 组件,rebuild 整个项目 (在 Android Studio 中,选择 Build > Rebuild Project)。

因为我们在覆盖默认的 Application 类,我们同样需要修改应用的 name 以启动 MyApp。这样,你的 application 将会使用这个 application 类来处理最初的实例化。

在我们的 activity 中,我们只需要获取这些 components 的引用,并调用 inject()

public class MyActivity extends Activity {  @Inject OkHttpClient mOkHttpClient;  @Inject SharedPreferences sharedPreferences;  public void onCreate(Bundle savedInstance) {        // assign singleton instances to fields        // We need to cast to `MyApp` in order to get the right method        ((MyApp) getApplication()).getNetComponent().inject(this);    }

限定词类型

1Dagger Qualifiers

如果我们需要同一类型的两个不同对象,我们可以使用 @Named 限定词注解。 你需要定义你如何提供单例 (用 @Provides 注解),以及你从哪里注入它们(用 @Inject 注解):

@Provides @Named("cached")@SingletonOkHttpClient provideOkHttpClient(Cache cache) {    OkHttpClient client = new OkHttpClient();    client.setCache(cache);    return client;}@Provides @Named("non_cached") @SingletonOkHttpClient provideOkHttpClient() {    OkHttpClient client = new OkHttpClient();    return client;}

注入同样需要这些 named 注解:

@Inject @Named("cached") OkHttpClient client;@Inject @Named("non_cached") OkHttpClient client2;

@Named 是一个被 Dagger 预先定义的限定语,但你也可以创建你自己的限定语注解:

@Qualifier@Documented@Retention(RUNTIME)public @interface DefaultPreferences {}

作用域

1Dagger 作用域

在 Dagger 2 中,你可以通过自定义作用域来定义组件应当如何封装。例如,你可以创建一个只持续 activity 或 fragment 整个生命周期的作用域。你也可以创建一个对应一个用户认证 session 的作用域。 你可以定义任意数量的自定义作用域注解,只要你把它们声明为 public@interface

@Scope@Documented@Retention(value=RetentionPolicy.RUNTIME)public @interface MyActivityScope{}

虽然 Dagger 2 在运行时不依赖注解,把 RetentionPolicy 设置为 RUNTIME 对于将来检查你的 module 将是很有用的。

依赖组件和子组件

利用作用域,我们可以创建 依赖组件 或 子组件。上面的例子中,我们使用了 @Singleton 注解,它持续了整个应用的生命周期。我们也依赖了一个主要的 Dagger 组件。

如果我们不需要组件总是存在于内存中(例如,和 activity 或 fragment 生命周期绑定,或在用户登录时绑定),我们可以创建依赖组件和子组件。它们各自提供了一种封装你的代码的方式。我们将在下一节中看到如何使用它们。

在使用这种方法时,有若干问题要注意:

  • 依赖组件需要父组件显式指定哪些依赖可以在下游注入,而子组件不需要 对父组件而言,你需要通过指定类型和方法来向下游组件暴露这些依赖:
// parent component@Singleton@Component(modules={AppModule.class, NetModule.class})public interface NetComponent {    // remove injection methods if downstream modules will perform injection    // downstream components need these exposed    // the method name does not matter, only the return type    Retrofit retrofit();     OkHttpClient okHttpClient();    SharedPreferences sharedPreferences();}

如果你忘记加入这一行,你将有可能看到一个关于注入目标缺失的错误。就像 private/public 变量的管理方式一样,使用一个 parent 组件可以更显式地控制,也可保证更好的封装。使用子组件使得依赖注入更容易管理,但封装得更差。

  • 两个依赖组件不能使用同一个作用域 例如,两个组件不能都用 @Singleton 注解设置定义域。这个限制的原因在  有所说明。依赖组件需要定义它们自己的作用域。

  • Dagger 2 同样允许使用带作用域的实例。你需要负责在合适的时机创建和销毁引用。Dagger 2 对底层实现一无所知。这个 Stack Overflow  上有更多的细节。

依赖组件

1Dagger 组件依赖

如果你想要创建一个组件,使它的生命周期和已登录用户的 session 相绑定,就可以创建UserScope 接口:

import java.lang.annotation.Retention;import javax.inject.Scope;@Scopepublic @interface UserScope {}

接下来,我们定义父组件:

@Singleton  @Component(modules={AppModule.class, NetModule.class})  public interface NetComponent {      // downstream components need these exposed with the return type      // method name does not really matter      Retrofit retrofit();  }

接下来定义子组件:

@UserScope // using the previously defined scope, note that @Singleton will not work@Component(dependencies = NetComponent.class, modules = GitHubModule.class)public interface GitHubComponent {    void inject(MainActivity activity);}

假定 Github 模块只是把 API 接口返回给 Github API:

@Modulepublic class GitHubModule {    public interface GitHubApiInterface {      @GET("/org/{orgName}/repos")      Call
> getRepository(@Path("orgName") String orgName); } @Provides @UserScope // needs to be consistent with the component scope public GitHubApiInterface providesGitHubInterface(Retrofit retrofit) { return retrofit.create(GitHubApiInterface.class); }}

为了让这个 GitHubModule.java 获得对 Retrofit 实例的引用,我们需要在上游组件中显式定义它们。如果下游模块会执行注入,它们也应当被从上游组件中移除:

@Singleton@Component(modules={AppModule.class, NetModule.class})public interface NetComponent {    // remove injection methods if downstream modules will perform injection    // downstream components need these exposed    Retrofit retrofit();    OkHttpClient okHttpClient();    SharedPreferences sharedPreferences();}

最终的步骤是用 GitHubComponent 进行实例化。这一次,我们需要首先实现 NetComponent 并把它传递给 DaggerGitHubComponent builder 的构造方法:

NetComponent mNetComponent = DaggerNetComponent.builder()                .appModule(new AppModule(this))                .netModule(new NetModule("https://api.github.com"))                .build();GitHubComponent gitHubComponent = DaggerGitHubComponent.builder()                .netComponent(mNetComponent)                .gitHubModule(new GitHubModule())                .build();

 中有一个实际的例子。

子组件

1Dagger 子组件

使用子组件是扩展组件对象图的另一种方式。就像带有依赖的组件一样,子组件有自己的的生命周期,而且在所有对子组件的引用都失效之后,可以被垃圾回收。此外它们作用域的限制也一样。使用这个方式的一个优点是你不需要定义所有的下游组件。

另一个主要的不同是,子组件需要在父组件中声明。

这是为一个 activity 使用子组件的例子。我们用自定义作用域和 @Subcomponent 注解这个类:

@MyActivityScope@Subcomponent(modules={ MyActivityModule.class })public interface MyActivitySubComponent {    @Named("my_list") ArrayAdapter myListAdapter();}

被使用的模块在下面定义:

@Modulepublic class MyActivityModule {    private final MyActivity activity;    // must be instantiated with an activity    public MyActivityModule(MyActivity activity) { this.activity = activity; }    @Provides @MyActivityScope @Named("my_list")    public ArrayAdapter providesMyListAdapter() {        return new ArrayAdapter
(activity, android.R.layout.my_list); } ...}

最后,在父组件中,我们将定义一个工厂方法,它以这个组件的类型作为返回值,并定义初始化所需的依赖:

@Singleton@Component(modules={ ... })public interface MyApplicationComponent {    // injection targets here    // factory method to instantiate the subcomponent defined here (passing in the module instance)    MyActivitySubComponent newMyActivitySubcomponent(MyActivityModule activityModule);}

在上面的例子中,一个子组件的新实例将在每次 newMyActivitySubcomponent() 调用时被创建。把这个子模块注入一个 activity 中:

public class MyActivity extends Activity {  @Inject ArrayAdapter arrayAdapter;  public void onCreate(Bundle savedInstance) {        // assign singleton instances to fields        // We need to cast to `MyApp` in order to get the right method        ((MyApp) getApplication()).getApplicationComponent())            .newMyActivitySubcomponent(new MyActivityModule(this))            .inject(this);    } }

子组件 builder

从 v2.7 版本起可用

1Dagger 子组件 builder

子组件 builder 使创建子组件的类和子组件的父类解耦。这是通过移除父组件中的子组件工厂方法实现的。

@MyActivityScope@Subcomponent(modules={ MyActivityModule.class })public interface MyActivitySubComponent {    ...    @Subcomponent.Builder    interface Builder extends SubcomponentBuilder
{ Builder activityModule(MyActivityModule module); }}public interface SubcomponentBuilder
{ V build();}

子组件是在子组件接口内部的接口中声明的。它必须含有一个 build() 方法,其返回值和子组件相匹配。用这个方法声明一个基接口是很方便的,就像上面的SubcomponentBuilder 一样。这个新的 builder 必须被加入父组件的图中,而这是用一个 "binder" 模块和一个 "subcomponents" 参数实现的:

@Module(subcomponents={ MyActivitySubComponent.class })public abstract class ApplicationBinders {    // Provide the builder to be included in a mapping used for creating the builders.    @Binds @IntoMap @SubcomponentKey(MyActivitySubComponent.Builder.class)    public abstract SubcomponentBuilder myActivity(MyActivitySubComponent.Builder impl);}@Component(modules={..., ApplicationBinders.class})public interface ApplicationComponent {    // Returns a map with all the builders mapped by their class.    Map
, Provider
> subcomponentBuilders();}// Needed only to to create the above mapping@MapKey @Target({ElementType.METHOD}) @Retention(RetentionPolicy.RUNTIME)public @interface SubcomponentKey { Class
value();}

一旦 builder 在出现在组件图中,activity 就可以用它来创建子组件:

public class MyActivity extends Activity {  @Inject ArrayAdapter arrayAdapter;  public void onCreate(Bundle savedInstance) {        // assign singleton instances to fields        // We need to cast to `MyApp` in order to get the right method        MyActivitySubcomponent.Builder builder = (MyActivitySubcomponent.Builder)            ((MyApp) getApplication()).getApplicationComponent())            .subcomponentBuilders()            .get(MyActivitySubcomponent.Builder.class)            .get();        builder.activityModule(new MyActivityModule(this)).build().inject(this);    } }

ProGuard

Dagger 2 应当在没有 ProGuard 时可以直接使用,但是如果你看到了 library class dagger.producers.monitoring.internal.Monitors$1 extends or implements program class javax.inject.Provider,你需要确认你的 gradle 配置使用了 annotationProcessor 声明,而不是 provided

常见问题

  • 如果你在升级 Dagger 版本(比如从 v2.0 升级到 v 2.5),一些被生成的代码会改变。如果你在集成使用旧版本 Dagger 生成的代码,你可能会看到 MemberInjector 和 actual and former argument lists different in length 错误。确保你 clean 过整个项目,并且把所有版本升级到和 Dagger 2 相匹配的版本。

参考资料

原文发布时间为:2017年6月19日
本文来自云栖社区合作伙伴掘金,了解相关信息可以关注掘金网站。

转载地址:http://gbszo.baihongyu.com/

你可能感兴趣的文章
[转][整理]Linux下找不到共享库(shared object)
查看>>
浅谈浏览器端JavaScript跨域解决方法
查看>>
Yii源码解读 - 事件
查看>>
同步与异步、异步与回调
查看>>
[Algo] Print Matrix Diagonal 对角打印
查看>>
近乎 5.3 发布,SNS 社区系统
查看>>
【直播回顾】蚂蚁金服高级开发工程师萧恺:IDEA 插件开发入门教程 ...
查看>>
全明星投资融资5亿美元,曾下注亏损百亿的滴滴,及商汤、陆金所等独角兽 ...
查看>>
(TPC-H测试 SF=10,SF=200) PostgreSQL 11 vs 10 vs Deepgreen
查看>>
Android中的 targetsdkversioin
查看>>
51Talk2019战略升级,发布互动教学产品妖果AI
查看>>
H3C交换机配件RS232配置线(DB9针转RJ45)
查看>>
防止sql注入攻击的方法总结
查看>>
putty自动登陆
查看>>
从周五开始美国服务器特别慢,美国服务器用户有同感吗? ...
查看>>
好程序员大数据技术分享:Zookeeper集群管理与选举 ...
查看>>
如何在线生成自己的短网址,可自定义域名 可统计点击数
查看>>
Oracle中的单行函数
查看>>
像在K8S集群中一样运行本地程序
查看>>
估算软件规模之前如何明确划分系统边界?
查看>>