Dagger 2 系列(六) -- 进阶篇:Component 依赖、@SubComponent 与多 Component 下的 Scope 使用限制

本文主要是介绍Dagger 2 系列(六) -- 进阶篇:Component 依赖、@SubComponent 与多 Component 下的 Scope 使用限制,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

Dagger2

  • 该系列博客的最终目标: 搭建 MVP + Dagger2 框架
  • 该系列博客包含以下几篇内容:
  1. Dagger 2 系列(一) – 前奏篇:依赖注入的基本介绍
  2. Dagger 2 系列(二) – 基础篇:@Inject、@Component
  3. Dagger 2 系列(三) – 基础篇:@Module 和@Provides
  4. Dagger 2 系列(四) – 基础篇:@Named 和 @Qualifier
  5. Dagger 2 系列(五) – 进阶篇:@Scope 和 @Singleton
  6. Dagger 2 系列(六) – 进阶篇:Component 依赖、@SubComponent 与多 Component 下的 Scope 使用限制

在这篇文章中你会看到什么:

  1. Module 依赖 是什么
  2. @SubComponent 是什么
  3. Module 依赖@SubComponent 不同
  4. Component 下的 Scope 的使用限制

通过该系列的前几篇博客的学习,应该基本掌握了 Dagger2 的基本概念、基本使用方法、Scope 的概念等,这些足够我们搭建一个简单 Demo 去了解 Dagger2 ,但是在正常的业务开发中仅仅了解这些就显得力不从心。

在这篇中我们就来了解一下 Component 依赖@SubComponent 的基本使用以及多 Compone 下的 Scope 的使用。

Component 依赖@SubComponent 均是在多 Component 下通过两者来组织 Component 的依赖关系,具体的依赖关系的建立是根据业务依赖建立。

1. Component 依赖

Component 依赖 是通过 @Component 的注解中 dependencies 选项来标识的,意思是指 该 Component 依赖 dependencies 指定的 Component

举个例子,看以下代码:

  • SubComponent
@Component(dependencies = AllComponent.class, modules = SubModule.class)
public interface SubComponent {void injectSubComponentActivity(ComponentDepActivity mDepActivity);
}

通过 dependencies = AllComponent.class 可知,SubComponent 依赖 AllComponent

  • AllComponent
@Component(modules = AllModule.class)
public interface AllComponent {AllBean getAllBean();
}

如果要想让依赖 AllComponentSubComponent 提供注入 AllModule 提供的 AllBean 实例对象,那么需要在 AllComponent 显式 的提供获得 AllBean 的方法,在本例中为 AllBean getAllBean(),否则将报出类似以下的错误:

错误: xxxx.AllBean cannot be provided without an @Inject constructor or from an @Provides- or @Produces-annotated method.
xxxx.ComponentDepActivity.mAllBean
[injected field of type: xxxx.AllBean mAllBean]

意思大致为 ComponentDepActivity 中不能正确的注入 AllBean 。

  • ComponentDepActivity
public class ComponentDepActivity extends AppCompatActivity {@InjectSubBean mSubBean;@InjectAllBean mAllBean;AllComponent mAllComponent;@Overrideprotected void onCreate(Bundle savedInstanceState) {super.onCreate(savedInstanceState);setContentView(R.layout.activity_component_dep);mAllComponent =  DaggerAllComponent.builder().allModule(new AllModule()).build();DaggerSubComponent.builder().allComponent(mAllComponent).build().injectSubComponentActivity(this);mSubBean.test();mAllBean.test();}
}
  • 打印日志
E/TAG: test :  SubBean
E/TAG: test :  AllBean

具体代码见: Dagger2Demo

2. @SubComponent

@SubComponent 也是管理 Component 间的依赖,不同的是这种方式不需要 在被依赖的 Component 中显式的声明可以获取相应类实例的方法。通过 @SubComponent 来管理的 Component 之间是一种 继承关系子 Component 理所当然的可以使用父 Component 的可以提供的类实例

具体使用如下:

  • SubComponent
@Subcomponent(modules = SubModule.class)
public interface SubComponent {void injectSubComponentActivity(ComponentSubActivity mDepActivity);
}
  • AllComponent
@Component(modules = AllModule.class)
public interface AllComponent {SubComponent addSub(SubModule mSubModule);
}

要在父 Component 中显式的声明 子 Component ,其具体格式为 :

子Component 方法名 (子Component 对应的 Module);
  • ComponentSubActivity
public class ComponentSubActivity extends AppCompatActivity {@InjectAllBean mAllBean;@InjectSubBean mSubBean;@Overrideprotected void onCreate(Bundle savedInstanceState) {super.onCreate(savedInstanceState);setContentView(R.layout.activity_component_sub);AllComponent mAllComponent = DaggerAllComponent.builder().build();mAllComponent.addSub(new SubModule()).injectSubComponentActivity(this);mAllBean.test();mSubBean.test();}
}
  • 运行结果
test :  AllBean
test :  SubBean

具体代码见:Dagger2Demo

3. Component 依赖@SubComponent 的不同

关于两者的不同 Dependency injection with Dagger 2 - Custom scopes 有以下解释:

In general we have two ways to do this - with @Subcomponent annotation or with Components dependencies. The main difference between them is an objects graph sharing. Subcomponents have access to entire objects graph from their parents while Component dependency gives access only to those which are exposed in Component interface.

大致意思如下:

Subcomponent 可以访问到 父 Component 的全部对象图,而 Component 只可以访问到在所依赖的 Component 中暴露出来的类。

我们看另外一段解释:

And what is more important all scoping stuff happens here. All instances taken from UserComponent inherited from AppComponent still are singletons (in Application scope). But those which are produced by UserModule (which is a part of UserComponent) will be “local singletons” which live as long as this UserComponent instance.

大致意思如下:

更重要的是,所有范围内的事情都发生在这里。从AppComponent继承的UserComponent获取的所有实例仍然是单例(在应用程序范围内)。但是由UserModule(它是UserComponent的一部分)生成的那些将是“本地单例”,其存在与此UserComponent实例一样长。

我们也可以这样概括:子组件可以访问到父组件中提供的实例,并且该实例的scope 和父组件定义的scope 相同

4. 多 Component 与 Scope 的使用

两者同时使用有以下限制:

  1. Component 和他所依赖的 Component 不能公用相同的 Scope,每个Component 都要有自己的 Scope,编译时会报错,因为这有可能破坏Scope的范围。
  2. @Singleton 的 Component 不能依赖其他 Component。正常来说使用 @Singleton 注解的Component应为全局的Component 。
  3. 无Scope的Component不能依赖有Scope的Component,因为这也会导致Scope 被破坏。
  4. Module 或通过构造函数注入依赖图的类和其 Component 必须使用相同的Scope 。

5. 总结

在日常的项目开发过程中,我们需要认真的考虑 Component 间的依赖关系,一般的,这些依赖关系的建立是根据具体的业务,就比如在 GitHubClient 项目中,UserComponent 依赖 AppComponentRepositoriesListActivityComponentRepositoryDetailsActivityComponent 依赖 UserComponent ,这就是按照业务线: 库列表库详情 是依赖于 用户 的,而 用户 是被 App 所管理。

在多 Component 下 Scope 的使用是有限制的, 其最终的标准都是不可以导致 Scope 被破坏。


参考资料

Dependency injection with Dagger 2 - the API

Dependency injection with Dagger 2 - Custom scopes

Dependency injection with Dagger 2 - Custom scopes

Dagger2:Scope、Component 间依赖和 @SubComponent

这篇关于Dagger 2 系列(六) -- 进阶篇:Component 依赖、@SubComponent 与多 Component 下的 Scope 使用限制的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/685449

相关文章

setsid 命令工作原理和使用案例介绍

《setsid命令工作原理和使用案例介绍》setsid命令在Linux中创建独立会话,使进程脱离终端运行,适用于守护进程和后台任务,通过重定向输出和确保权限,可有效管理长时间运行的进程,本文给大家介... 目录setsid 命令介绍和使用案例基本介绍基本语法主要特点命令参数使用案例1. 在后台运行命令2.

使用Redis快速实现共享Session登录的详细步骤

《使用Redis快速实现共享Session登录的详细步骤》在Web开发中,Session通常用于存储用户的会话信息,允许用户在多个页面之间保持登录状态,Redis是一个开源的高性能键值数据库,广泛用于... 目录前言实现原理:步骤:使用Redis实现共享Session登录1. 引入Redis依赖2. 配置R

使用Python的requests库调用API接口的详细步骤

《使用Python的requests库调用API接口的详细步骤》使用Python的requests库调用API接口是开发中最常用的方式之一,它简化了HTTP请求的处理流程,以下是详细步骤和实战示例,涵... 目录一、准备工作:安装 requests 库二、基本调用流程(以 RESTful API 为例)1.

使用Python开发一个Ditto剪贴板数据导出工具

《使用Python开发一个Ditto剪贴板数据导出工具》在日常工作中,我们经常需要处理大量的剪贴板数据,下面将介绍如何使用Python的wxPython库开发一个图形化工具,实现从Ditto数据库中读... 目录前言运行结果项目需求分析技术选型核心功能实现1. Ditto数据库结构分析2. 数据库自动定位3

Python yield与yield from的简单使用方式

《Pythonyield与yieldfrom的简单使用方式》生成器通过yield定义,可在处理I/O时暂停执行并返回部分结果,待其他任务完成后继续,yieldfrom用于将一个生成器的值传递给另一... 目录python yield与yield from的使用代码结构总结Python yield与yield

Go语言使用select监听多个channel的示例详解

《Go语言使用select监听多个channel的示例详解》本文将聚焦Go并发中的一个强力工具,select,这篇文章将通过实际案例学习如何优雅地监听多个Channel,实现多任务处理、超时控制和非阻... 目录一、前言:为什么要使用select二、实战目标三、案例代码:监听两个任务结果和超时四、运行示例五

python使用Akshare与Streamlit实现股票估值分析教程(图文代码)

《python使用Akshare与Streamlit实现股票估值分析教程(图文代码)》入职测试中的一道题,要求:从Akshare下载某一个股票近十年的财务报表包括,资产负债表,利润表,现金流量表,保存... 目录一、前言二、核心知识点梳理1、Akshare数据获取2、Pandas数据处理3、Matplotl

Java使用Thumbnailator库实现图片处理与压缩功能

《Java使用Thumbnailator库实现图片处理与压缩功能》Thumbnailator是高性能Java图像处理库,支持缩放、旋转、水印添加、裁剪及格式转换,提供易用API和性能优化,适合Web应... 目录1. 图片处理库Thumbnailator介绍2. 基本和指定大小图片缩放功能2.1 图片缩放的

Python使用Tenacity一行代码实现自动重试详解

《Python使用Tenacity一行代码实现自动重试详解》tenacity是一个专为Python设计的通用重试库,它的核心理念就是用简单、清晰的方式,为任何可能失败的操作添加重试能力,下面我们就来看... 目录一切始于一个简单的 API 调用Tenacity 入门:一行代码实现优雅重试精细控制:让重试按我

MySQL中EXISTS与IN用法使用与对比分析

《MySQL中EXISTS与IN用法使用与对比分析》在MySQL中,EXISTS和IN都用于子查询中根据另一个查询的结果来过滤主查询的记录,本文将基于工作原理、效率和应用场景进行全面对比... 目录一、基本用法详解1. IN 运算符2. EXISTS 运算符二、EXISTS 与 IN 的选择策略三、性能对比