文档章节

Dagger2图文完全教程

代码GG
 代码GG
发布于 2016/06/28 19:35
字数 1937
阅读 126
收藏 4

#Dagger2图文完全教程


本文属代码GG原创,非经本人同意,禁止转载。

需要交流,联系微信:code_gg_boy 更多精彩,时时关注微信公众号code_gg_home

一个显示图

没有更多开场白,直接说下我对它的理解。

Dagger2 是一个Android依赖注入框架。而android开发当前非常流行的非MVP模式莫属了,Dagger2的目标便是将MVP中的V P 进一步解耦,达到模块化最大的解耦,使得代码更容易维护。

举个栗子:有个A对象 B对象 和C对象,如果C对象创建需要A和B,那么我们是不是需要构造里面传入参数A和参数B,然后在使用的地方如下写个代码:

C c=new C(new A(),new B());

如果我们使用了Dagger2时候,我们就不需要管这些了,只需要关联住能提供创建A 和 B的地方 ,然后在需要C的地方写下:

@Inject
C c;

然后在这个类的初始化地方进行注入即可。 我们初步来看,会发现Dagger2优势不大,没什么吸引人的,那么请你静下心来,看完再得出结论。


闲话休叙,我们来直接上代码:(常规写法)

#1 编写一个类:

public class Test3 {
    public Test3() {
    }
}

#2 使用的地方

public class MainActivity extends AppCompatActivity {
    Test3 test3;
    @Override
    protected void onCreate(Bundle savedInstanceState) {
        //.....
        test3 = new Test3();
    }
}

  • 如果我们改为使用Dagger2的方式的话,则可以写成如下方式:

#1 创建一个类

使用了注解方式,使得Dagger2能找到它。

public class Test3 {
    //这里可以看到加入了注解方式
    @Inject
    public Test3() {
    }
}

#2 新增一个对象:

@Singleton
//用这个标注标识是一个连接器
@Component()
public interface MainActivityComponent {
    //这个连接器要注入的对象。这个inject标注的意思是,我后面的参数对象里面有标注为@Inject的属性,这个标注的属性是需要这个连接器注入进来的。
    void inject(MainActivity activity);
}

#3 调用的地方改为:

public class MainActivity extends AppCompatActivity {
    //加入注解,标注这个test3是需要注入的
    @Inject
    Test3 test3;
    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        //使用组件进行构造,注入
        DaggerMainActivityComponent.builder().build().inject(this);
    }

这是最简单的一种使用了。首先我们看到,第一印象是我去,这个更复杂了啊。我只能说确实,因为这个是它对的最基础的使用,看起来很笨拙,但是当它在大型项目里面,在依赖更多的情况下,则会发生质的飞跃,会发现它非常好用,并且将你需要传递的参数都隐藏掉,来实现解耦。


我先说下Dagger2的注释思路:关键的点是@Component,这个是个连接器,用来连接提供方和使用方的,所以它是桥梁。它使用在组件里面标记使用的Module(标记用到了哪个Module,主要是看使用方需要哪些对象进行构造,然后将它的提供方@module写在这里) 然后我们写入一个void inject(MainActivity activity); 这里后面的参数,就是我们的使用方了。如此一来,我们在使用的地方,使用类似这种方式(DaggerMainActivityComponent.builder().build().inject(this);)的动作,将使用方类里面的标记 为@Inject的类初始化掉,完成自动初始化的动作。

结构如下:

一个显示图

为了更好的来学习它,我们来依次看看各种使用情况。

#1 常规使用方法

一个显示图 一个显示图 一个显示图

直接感受下,如何?

#2 带一个参数的效果


一个显示图


一个显示图


一个显示图


我们来看一个代码段,当我们创建两个实例的时候,发现地址是独立的。

一个显示图

如果我们想要一样的地址呢?加上一句话,具体如下:

一个显示图

效果便是两个共用实例啦。

一个显示图

#3 换种经常使用的方式

将提供的构造,放入@module里面,具体效果如下:

一个显示图

去掉标记的@singleton后

一个显示图

效果变成独立的啦

一个显示图

#4 依赖一个组件的时候

有时我们需要依赖一个组件,这个最常见的用法是,我们App实例里面提供了比如获取sharepreference的实例,和比如现在代码里面的LocationManager的实例,我们Activity里面需要这些实例,我们该如何来做呢?看效果: 1:一个AndroidModule 模块标记

一个显示图

这个模块属于AndroidcationComponent 组件里面

一个显示图

这里有个关键点,就是子组件需要这个里面的某个实例的时候,这里需要使用一个接口,将需要的实例做一个返回动作。这里是LocationManager这一行。

我们的子组件的代码如下:

一个显示图

对应的Cmodule代码如下:

一个显示图

再来看下Test3的代码当前情况:

一个显示图

使用的地方:

一个显示图


细心的你会发现这里多了一个注释了,@PerActivity,它是个什么鬼呢?

一个显示图

这里我们看到它是使用了@Scope的一个注释,这个注释的意思就是作用域,在作用域内保持单例,可以直接理解为单例即可。 为什么要新增一个呢,主要是因为各个组件需要独立出来,因此如果是依赖关系,则需要各自在不同的注释作用域里面。 我们来看下在Cmodule里面,加上@perActivity注释后的效果:

一个显示图

一个显示图

如果去掉呢?

一个显示图

我们突然发现,它和单例的注释起的作用一样啊。so。。。是不是发现什么啦。

一个显示图

因此我们得出一个结论,这里@Singleton 就是一个普通的作用域通道,使用了作用域@Scope注释的代码,会变成单例模式。为了验证我们的思路,作如下测试:

一个显示图

一个显示图

我们将之前的@Singleton用新建的这个替换掉,验证两次的生成代码,发现一模一样,一模一样,一模一样,so。。。 就是这个样子啦。

#5 自定义一个标记 为什么要自定义标记呢?这个标记不是使用@Scope注释的哦,是使用@Qualifier 标记的,它的目标是,为了区分如果同时返回类型一样,比如构造男孩,女孩的基本属性,性别和名字时候,获取男孩和女孩都是一个对象,我们该如何区分呢,这个就是关键啦。说这么多,真心很烦,直接栗子来啦。

这里稍安勿躁,先来看相同效果的另一个注释,@Name,这个是Dagger2自带的一个让区分,效果如下:

一个显示图

一个显示图

一个显示图

一个显示图

这里@Name可以简单的一个使用方式,就是它不是区分对象,而是限制使用时候必须加入这个注释,否则报错,目的就是让使用者注意是否使用正确了。

一个显示图

一个显示图

我们使用自己的注释再来一遍:

一个显示图

一个显示图

一个显示图

一个显示图

一个显示图

对比两种方式,我们发现使用@Name的时候,后面的注释名字会敲错,而我们第二种方式呢,则不会耶,so。。。

我们看下自定义的标记,作为限制出错,让强制标注的例子。

一个显示图

一个显示图

一个显示图

一个显示图

#6 子组件(公共组件) 这个出现的目的是为了如果有一个组件,是每次创建实例提供给别人,而恰好其他组件(有多个)里面有需要它,如果只有一个,我们就用依赖搞定啦。那么它就可以定义成子组件,谁需要在谁的组件里面加一下,具体看例子:

一个显示图

一个显示图

一个显示图

如上,写完啦。。

实战地方,可以参照 https://github.com/gzsll/TLint 来阅读啦,收工。 需要交流,联系微信:code_gg_boy 更多精彩,时时关注微信公众号code_gg_home

一个显示图

© 著作权归作者所有

代码GG
粉丝 0
博文 1
码字总数 1937
作品 0
西安
私信 提问
Android databinding和dagger2冲突问题

databinding和dagger2之间还会因为android-apt出现冲突,单独使用databinding,编译能通过,同时使用dagger2,databinding就不能生成相应的类,当前dagger2版本为2.9 尝试过网上的教程:更新...

YinLinTang
2017/04/10
197
0
爽翻天!告别Java。一起来使用kotlin开发完整客户端

必须写在前面:从Java转向Kotlin,只需要几天时间! 本篇是kotlin开发Android系列教程的最后一篇。前面几篇介绍了kotlin的基本语法、与java的不同之处等。在本篇文章中,将使用到前面的知识点...

wingichoy
2016/11/28
0
0
Dagger2的使用

想学习下Dagger2 ,搜了很多资料但都是附带着RxJava等其他东西,让楼主学习起来比较费时(可能比较笨),本人也没有使用过Dagger1.x,所以学习Dagger2的更是举步维艰啊。 看了很多资料,终于...

zhx2012
2016/07/04
98
0
dagger2 让你爱不释手:终结篇

本文作者: 伯乐在线 - 牛犇 。未经作者许可,禁止转载! 欢迎加入伯乐在线作者团队。 前言 如果您对dagger2的概念,整个依赖注入框架还不清楚,可以先了解下我的前2篇文章: 《dagger2让你爱...

刘星石
2016/04/01
276
0
Dagger2让你爱不释手-基础依赖注入框架篇

前言 dagger2的大名我想大家都已经很熟了,它是解决Android或java中依赖注入的一个类库(DI类库)。当我看到一些开源的项目在使用dagger2时,我也有种匆匆欲动的感觉,因此就立马想一探它的究...

技术小能手
2018/07/10
0
0

没有更多内容

加载失败,请刷新页面

加载更多

分布式协调服务zookeeper

ps.本文为《从Paxos到Zookeeper 分布式一致性原理与实践》笔记之一 ZooKeeper ZooKeeper曾是Apache Hadoop的一个子项目,是一个典型的分布式数据一致性的解决方案,分布式应用程序可以基于它...

ls_cherish
今天
4
0
redis 学习2

网站 启动 服务端 启动redis 服务端 在redis 安装目录下 src 里面 ./redis-server & 可以指定 配置文件或者端口 客户端 在 redis 的安装目录里面的 src 里面 ./redis-cli 可以指定 指定 连接...

之渊
昨天
2
0
Spring boot 静态资源访问

0. 两个配置 spring.mvc.static-path-patternspring.resources.static-locations 1. application中需要先行的两个配置项 1.1 spring.mvc.static-path-pattern 这个配置项是告诉springboo......

moon888
昨天
4
0
hash slot(虚拟桶)

在分布式集群中,如何保证相同请求落到相同的机器上,并且后面的集群机器可以尽可能的均分请求,并且当扩容或down机的情况下能对原有集群影响最小。 round robin算法:是把数据mod后直接映射...

李朝强
昨天
4
0
Kafka 原理和实战

本文首发于 vivo互联网技术 微信公众号 https://mp.weixin.qq.com/s/bV8AhqAjQp4a_iXRfobkCQ 作者简介:郑志彬,毕业于华南理工大学计算机科学与技术(双语班)。先后从事过电子商务、开放平...

vivo互联网技术
昨天
24
0

没有更多内容

加载失败,请刷新页面

加载更多

返回顶部
顶部