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

相关文章

Spring Security 从入门到进阶系列教程

Spring Security 入门系列 《保护 Web 应用的安全》 《Spring-Security-入门(一):登录与退出》 《Spring-Security-入门(二):基于数据库验证》 《Spring-Security-入门(三):密码加密》 《Spring-Security-入门(四):自定义-Filter》 《Spring-Security-入门(五):在 Sprin

中文分词jieba库的使用与实景应用(一)

知识星球:https://articles.zsxq.com/id_fxvgc803qmr2.html 目录 一.定义: 精确模式(默认模式): 全模式: 搜索引擎模式: paddle 模式(基于深度学习的分词模式): 二 自定义词典 三.文本解析   调整词出现的频率 四. 关键词提取 A. 基于TF-IDF算法的关键词提取 B. 基于TextRank算法的关键词提取

使用SecondaryNameNode恢复NameNode的数据

1)需求: NameNode进程挂了并且存储的数据也丢失了,如何恢复NameNode 此种方式恢复的数据可能存在小部分数据的丢失。 2)故障模拟 (1)kill -9 NameNode进程 [lytfly@hadoop102 current]$ kill -9 19886 (2)删除NameNode存储的数据(/opt/module/hadoop-3.1.4/data/tmp/dfs/na

Hadoop数据压缩使用介绍

一、压缩原则 (1)运算密集型的Job,少用压缩 (2)IO密集型的Job,多用压缩 二、压缩算法比较 三、压缩位置选择 四、压缩参数配置 1)为了支持多种压缩/解压缩算法,Hadoop引入了编码/解码器 2)要在Hadoop中启用压缩,可以配置如下参数

Makefile简明使用教程

文章目录 规则makefile文件的基本语法:加在命令前的特殊符号:.PHONY伪目标: Makefilev1 直观写法v2 加上中间过程v3 伪目标v4 变量 make 选项-f-n-C Make 是一种流行的构建工具,常用于将源代码转换成可执行文件或者其他形式的输出文件(如库文件、文档等)。Make 可以自动化地执行编译、链接等一系列操作。 规则 makefile文件

每天认识几个maven依赖(ActiveMQ+activemq-jaxb+activesoap+activespace+adarwin)

八、ActiveMQ 1、是什么? ActiveMQ 是一个开源的消息中间件(Message Broker),由 Apache 软件基金会开发和维护。它实现了 Java 消息服务(Java Message Service, JMS)规范,并支持多种消息传递协议,包括 AMQP、MQTT 和 OpenWire 等。 2、有什么用? 可靠性:ActiveMQ 提供了消息持久性和事务支持,确保消

使用opencv优化图片(画面变清晰)

文章目录 需求影响照片清晰度的因素 实现降噪测试代码 锐化空间锐化Unsharp Masking频率域锐化对比测试 对比度增强常用算法对比测试 需求 对图像进行优化,使其看起来更清晰,同时保持尺寸不变,通常涉及到图像处理技术如锐化、降噪、对比度增强等 影响照片清晰度的因素 影响照片清晰度的因素有很多,主要可以从以下几个方面来分析 1. 拍摄设备 相机传感器:相机传

pdfmake生成pdf的使用

实际项目中有时会有根据填写的表单数据或者其他格式的数据,将数据自动填充到pdf文件中根据固定模板生成pdf文件的需求 文章目录 利用pdfmake生成pdf文件1.下载安装pdfmake第三方包2.封装生成pdf文件的共用配置3.生成pdf文件的文件模板内容4.调用方法生成pdf 利用pdfmake生成pdf文件 1.下载安装pdfmake第三方包 npm i pdfma

零基础学习Redis(10) -- zset类型命令使用

zset是有序集合,内部除了存储元素外,还会存储一个score,存储在zset中的元素会按照score的大小升序排列,不同元素的score可以重复,score相同的元素会按照元素的字典序排列。 1. zset常用命令 1.1 zadd  zadd key [NX | XX] [GT | LT]   [CH] [INCR] score member [score member ...]

科研绘图系列:R语言扩展物种堆积图(Extended Stacked Barplot)

介绍 R语言的扩展物种堆积图是一种数据可视化工具,它不仅展示了物种的堆积结果,还整合了不同样本分组之间的差异性分析结果。这种图形表示方法能够直观地比较不同物种在各个分组中的显著性差异,为研究者提供了一种有效的数据解读方式。 加载R包 knitr::opts_chunk$set(warning = F, message = F)library(tidyverse)library(phyl