6.绝绝子--Android - 抖音使用Dagger2实现MVP+RecycleView解藕 ( 依赖注入一看就明白) 没看过你就Out了

dagger的用途就是:让你不需要初始化对象。换句话说,任何对象声明完了就能直接用。

依赖注入的好处

重用 Car,依赖通过外部传入过来!

`* 重用类以及分离依赖项:更容易换掉依赖项的实现。由于控制反转,代码重用得以改进,并且类不再控制其依赖项的创建方式,而是支持任何配置。

  • 易于重构:依赖项成为 API Surface 的可验证部分,因此可以在创建对象时或编译时进行检查,而不是作为实现详情隐藏。`

目的,改造演变:

第一种:直接new ,很多个地方new

第二种:统一new ,通过工厂的方式new,以后只改一个地方就可以

第三种:统一的地方,自动new

如何解藕?

1.MVP + Dagger2

专为MVP模式中的P层和V层做进一步解耦

因为在MVP模式中Activity持有presenter的引用,同时presenter也持有view的引用,这样便于更新UI界面,这样Activity就和 presenter仅仅的耦合在一起了,而Dagger2是依赖注入框架就是解耦合的,所以子MVP中使用Dagger2也就再好不过了。

activity持有了presenter的引用并且创建了该对象,但是如果presenter的构造函数发生改变则这里也需要改变,其实所有和presenter构造函数相关的代码都要改变。 

把很多对象new的地方统一起来了。之前用的是工厂模式的方式!工厂比直接new好一点,dagger2是自动注册,自动手写了很多new

我的理解:

1).MVP中 present还是会有view的依赖!

                如果很多个地方使用这个presenter。只要改一个地方就可以了,不用改多个地方。

2). 另外一种情况:

     A里面有B, B里面有c,C里面有D。

如果都手动new ,如果后面改起来,就会很麻烦!

2.dagger2+组件化


一:Dagger2是什么?

是一个依赖注入框架,butterknife也是一个依赖注入框架。不过butterknife,最多叫奶油刀,Dagger2被叫做利器啊,他的主要作用,就是对象的管理

优点:

其目的是为了降低程序耦合。

随即而来的就是可测试性,可维护,可扩展性就大大提高了

很多小白对于Dagger2是啥浑然不知,更不知其能带来的好处了。

这里举个例子,比如有个类A,他的构造函数需要传入B,C;然后代码里有10个地方实例化了A,那如果功能更改,A的构造函数改成了只有B,这个时候,你是不是要去这10个地方一个一个的改?如果是100个地方,你是不是要吐血?!如果采用dagger2,这样的需求只需要改1-2个地方,你感觉怎么样?对你有诱惑吗?

原理:

这类依赖注入框架都已经采用了apt代码自动生成技术,其注解是停留在编译时,完全不影响性能。

dagger是使用依赖注入的方式,使用Annotation给需要注入的对象做标记,通过inject()方法自动注入所有对象,从而完成自动的初始化

dagger2 解决什么问题

第一:dagger 是一个依赖注入框架,首要任务当然是解决依赖注入的问题。
第二:dagger主要想通过编译时产生代码的方式来解决那些基于反射的依赖注入框架所存在的缺点,例如性能问题,开发过程中存在的问题。

注入的形式:

<pre style="margin: 8px 0px; font-style: normal; font-variant-ligatures: normal; font-variant-caps: normal; font-weight: 400; letter-spacing: normal; orphans: 2; text-align: start; text-indent: 0px; text-transform: none; widows: 2; word-spacing: 0px; -webkit-text-stroke-width: 0px; text-decoration-thickness: initial; text-decoration-style: initial; text-decoration-color: initial; background-color: rgb(43, 43, 43); color: rgb(169, 183, 198); font-family: 宋体; font-size: 0.8rem;">public class MainActivity extends AppCompatActivity {

@Named("dev")
@Inject
MainApi apiDev;    @Named("release")
@Inject
MainApi apiRelease;    @Override

protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main); DaggerMainComponent.builder()
.mainModule(new MainModule(this))
.mainChildModule(new MainChildModule())
.build()
.inject(this);
apiDev.eat();
apiRelease.eat();
Log.i("TAG","apiDev--->" + apiDev);
Log.i("TAG","apiRelease--->" + apiRelease);
}</pre>

3要素:Inject Component Module

@Inject

在A类中标记B类的字段:告诉dagger 我们要注入B类的实例到A类中,你帮我搞定。

在B类中标记B类的构造函数:告诉dagger B类是一个可以被注入的类,如果你想把B的实例对象注入到其他类中,例如注入到A类中,是没有任何问题的。

@Component :中间桥接,对外接口

一般是一个使用 @Component标记的接口,其持有A类的实例,其在A类中发现有使用@Inject标记的属性b

@Module和 providers绑定在一起

C类是一个没有使用@Inject注解其构造函数的类

@Module注解表示,这个类是一个Module,Module的作用是提供信息,让ObjectGraph知道应该怎样注入所有的依赖

3、Component依赖Component

1、最简单不带ModuleInject方式

由我们自己定义的类,我们可以自由修改的情况下我们使用这种方式,也分为两种:带参数和不带参数的。

a、构造参数不带参数的:

例子:

b、构造参数带参数的:

构造函数如果带了inject注入的话,类就默认被注入了,不用像无惨的一样

例子:

发现Factory的构造函数被@Inject标注了且带有一个参数,

然后dagger2就去寻找Product发现它的构造函数也被@Inject标注并且无参数,于是dagger2把Product的实例注入给FactoryActivity

2、带ModuleInject方式

应用场景:若是我们引入的第三方库不能随意改动代码的话就不方便了,我们这里使用如下RetrofitManager模拟不可改动代码的情况:

  • 第三方库提供的类,它们的构造方法不能被注解

对于这样的情况,可以使用@Provides注解来提供专用的初始化方法,实现自定义依赖。

@Provides
Coder provideCoder(Boss boss) {
    return new Coder(boss);
}

include的用法

<pre style="margin: 8px 0px; background-color: rgb(43, 43, 43); color: rgb(169, 183, 198); font-family: 宋体; font-size: 0.8rem;">@Module (includes = {BModule.class})// includes 引入) public class AModule {
@Provides
A providerA() {
return new A();
}
}</pre>

@Named注解用

相当于有个表示,虽然大家都是同一个对象,但是实例化对象不同就不如

 A a1 = new A();    A a2 = new A();// a1  a2 能一样嘛

Module中 使用@Named注解

@Module
public class MainModule {

    private MainActivity activity;

    public MainModule(MainActivity activity) {
        this.activity = activity;
    }

    @Named("dev")
    @Provides
    MainApi provideMainApiDev(MainChildApi mainChildApi, String url) {
        return new MainApi(mainChildApi, activity,"dev");
    }

    @Named("release")
    @Provides
    MainApi provideMainApiRelease(MainChildApi mainChildApi, String url) {
        return new MainApi(mainChildApi, activity,"release");
    }

}来源: http://08643.cn/p/2cd491f0da01

在Activity/Fragment中使用

<pre class="hljs java" style="margin: 8px 0px;">

<pre style="margin: 8px 0px; background-color: rgb(43, 43, 43); color: rgb(169, 183, 198); font-family: 宋体; font-size: 0.8rem;">public class MainActivity extends AppCompatActivity {

@Named("dev")
@Inject
MainApi apiDev;    @Named("release")
@Inject
MainApi apiRelease;    @Override

protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main); DaggerMainComponent.builder()
.mainModule(new MainModule(this))
.mainChildModule(new MainChildModule())
.build()
.inject(this);
apiDev.eat();
apiRelease.eat();
Log.i("TAG","apiDev--->" + apiDev);
Log.i("TAG","apiRelease--->" + apiRelease);
}

}</pre>

</pre>

<pre class="hljs java" style="margin: 8px 0px;">在项目中的应用场景和实例:
</pre>

<pre class="hljs java" style="margin: 8px 0px;">https://blog.csdn.net/soslinken/article/details/52184113</pre>

注意:

实质上,Dagger会在编译时对代码进行检查,并在检查不通过的时候报编译错误(为什么?这和Dagger的原理有关,有兴趣的话可以关注我之后发布的Dagger详解)。检查内容主要有三点:

  1. 所有含有依赖注入的类,需要被显式 声明在相应的Module中。
  2. 一个Module中所有@Provides方法的参数都必须在这个Module种提供相应的@Provides方法,或者在@Module注解后添加“complete = false”注明这是一个不完整Module(即它会被其他Module所扩展)。
  3. 一个Module中所有的@Provides方法都要被它声明的注入对象所使用,或者在@Module注解后添加“library = ture”注明(即它是为了扩展其他Module而存在的)。

@Binds:的使用。

————————————————

总结:

1.作用域--------重点,比较难的地方!

2.组件依赖

3.子组件 subConponent

爬坑指南(极度重要)

  1. Provide 如果是单例模式 对应的Compnent 也要是单例模式
  2. inject(Activity act) 不能放父类
  3. 即使使用了单利模式,在不同的Activity 对象还是不一样的
  4. 依赖component, component之间的Scoped 不能相同
  5. 子类component 依赖父类的component ,子类component的Scoped 要小于父类的Scoped,Singleton的级别是Application
  6. 多个Moudle 之间不能提供相同的对象实例
  7. Moudle 中使用了自定义的Scoped 那么对应的Compnent 使用同样的Scoped

重点是:处理作用域provider

通过APT+javapoAT在编译时候生成上面3个类!

APT的process()方法是什么时候执行?

原理分析:

1.a是怎么实例化的

2.b和c是怎么是咧化的

3.他们是怎么绑定的

原理:在build里面生成了很多的类,工厂模式,代理模式,A_Factory ,proxyProviderC

单列如何实现?

保持conponent是全局唯一的,通过application中初始化!

Dragger的原理:

[图片上传失败...(image-115c54-1640679669730)]

1.DaggerPengConponnet------注入和对象的关系

在里面,通过工厂模式得到provider

就是下面要用的2个类PengModule_GetHttpObjectFactory和MainActivity_MembersInjector

<pre style="margin: 8px 0px; background-color: rgb(43, 43, 43); color: rgb(169, 183, 198); font-family: "JetBrains Mono", monospace; font-size: 0.817rem;">public final class DaggerPengConponnet implements PengConponnet {
private Provider<HttpObject> getHttpObjectProvider; private MembersInjector<MainActivity> mainActivityMembersInjector; private DaggerPengConponnet(Builder builder) {
assert builder != null;
initialize(builder);
}

public static Builder builder() {
return new Builder();
}

public static PengConponnet create() {
return new Builder().build();
}

@SuppressWarnings("unchecked")
private void initialize(final Builder builder) {

this.getHttpObjectProvider = PengModule_GetHttpObjectFactory.create(builder.pengModule);   this.mainActivityMembersInjector = MainActivity_MembersInjector.create(getHttpObjectProvider);

}

@Override
public void injectActivity(MainActivity mainActivity) {
mainActivityMembersInjector.injectMembers(mainActivity);
}

public static final class Builder {
private PengModule pengModule; private Builder() {}

public PengConponnet build() {
  if (pengModule == null) {
    this.pengModule = new PengModule();

}
return new DaggerPengConponnet(this);
}

public Builder pengModule(PengModule pengModule) {
  this.pengModule = Preconditions.checkNotNull(pengModule);

return this; }
}
}
</pre>

2.PengModule_GetHttpObjectFactory --------工厂,new出对象。

通过工厂模式创建对象

<pre style="margin: 8px 0px; background-color: rgb(43, 43, 43); color: rgb(169, 183, 198); font-family: "JetBrains Mono", monospace; font-size: 0.817rem;">public final class PengModule_GetHttpObjectFactory implements Factory<HttpObject> {
private final PengModule module; public PengModule_GetHttpObjectFactory(PengModule module) {
assert module != null;
this.module = module;
}

@Override
public HttpObject get() {
return Preconditions.checkNotNull(
module.getHttpObject(), "Cannot return null from a non-@Nullable @Provides method");
}

public static Factory<HttpObject> create(PengModule module) {
return new PengModule_GetHttpObjectFactory(module);
}
}3.MainActivity_MembersInjector -----赋值</pre>

通过provider得到了需要注入的对象

3.MainActivity_MembersInjector:注入绑定

<pre style="margin: 8px 0px; background-color: rgb(43, 43, 43); color: rgb(169, 183, 198); font-family: "JetBrains Mono", monospace; font-size: 0.817rem;">public final class MainActivity_MembersInjector implements MembersInjector<MainActivity> {
private final Provider<HttpObject> httpObjectProvider; public MainActivity_MembersInjector(Provider<HttpObject> httpObjectProvider) {
assert httpObjectProvider != null;
this.httpObjectProvider = httpObjectProvider;
}

public static MembersInjector<MainActivity> create(Provider<HttpObject> httpObjectProvider) {
return new MainActivity_MembersInjector(httpObjectProvider);
}

@Override
public void injectMembers(MainActivity instance) {
if (instance == null) {
throw new NullPointerException("Cannot inject members into a null reference");
}
instance.httpObject = httpObjectProvider.get();
}

public static void injectHttpObject(
MainActivity instance, Provider<HttpObject> httpObjectProvider) {
instance.httpObject = httpObjectProvider.get();
}
}</pre>

20211227最新使用:

[图片上传失败...(image-cef848-1640679669730)]

使用4部曲

现在用Dagger2来改造,总体来说就是4步:

  • 1 在Adapter构造器中加入@Inject注解(告诉他,如何创建对象)
  • 2 构建 Module(不一定需要)
  • 3 构建 Component(声明是dagger的组件,传入的参数是注入到哪里去)
  • 4 完成依赖注入(调用完成注入)

在操作中会使用到了@Inject、@Module、@Provides、@Conponent注解,那么他们分别在完成什么工作?

MainActivityPresenter presenter = new MainActivityPresenter(this);

@Inject @Conponent @Module @Provides

这四个注解可以这么理解,@Inject承担了=号左边的工作,@Conponent 承担了=号的工作, @Module @Provides承担了=号右边的工作

自己的步骤理解:

1.本来就有的bean对象获取其他对象httpObject

2.定义module:用来装载上面的object,一个module可以有多个object

3.定义组件component: 用来装module,一个component可以有多个module,一般是一个module.

4.使用,在activty中,直接定义httpobject,然后oncreate通过注解注入就可以了。

具体代码:

[图片上传失败...(image-8e9472-1640679669730)]

第一步: Bean

<pre style="margin: 8px 0px; background-color: rgb(43, 43, 43); color: rgb(169, 183, 198); font-family: "JetBrains Mono", monospace; font-size: 0.817rem;">public class HttpObject {
}
</pre>

第二步:Module

<pre style="margin: 8px 0px; background-color: rgb(43, 43, 43); color: rgb(169, 183, 198); font-family: "JetBrains Mono", monospace; font-size: 0.817rem;">@Module public class PengModule {

@Provides

public HttpObject getHttpObject() {
return new HttpObject();
}

@Provides

public DatabaseObject getDatabase() {
return new DatabaseObject();
}

}
</pre>

第三步:Componet

<pre style="margin: 8px 0px; background-color: rgb(43, 43, 43); color: rgb(169, 183, 198); font-family: "JetBrains Mono", monospace; font-size: 0.817rem;">@Component(modules = PengModule.class) //可以有多个module public interface PengConponnet { // 接口给外部调用
void injectActivity(MainActivity mainActivity); }
</pre>

第四步:使用,需要用自动生成的类

<pre style="margin: 8px 0px; background-color: rgb(43, 43, 43); color: rgb(169, 183, 198); font-family: "JetBrains Mono", monospace; font-size: 0.817rem;">public class MainActivity extends AppCompatActivity {

@Inject

HttpObject httpObject; @Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main); //自动生成类进行注入
DaggerPengConponnet.create().injectActivity(this);
}
}
</pre>

非常不错:

http://08643.cn/p/2cd491f0da01

?著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 214,029评论 6 493
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,238评论 3 388
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事?!?“怎么了?”我有些...
    开封第一讲书人阅读 159,576评论 0 349
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,214评论 1 287
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,324评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,392评论 1 292
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,416评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,196评论 0 269
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,631评论 1 306
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,919评论 2 328
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,090评论 1 342
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,767评论 4 337
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,410评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,090评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,328评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,952评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,979评论 2 351

推荐阅读更多精彩内容

  • Dagger2入门详解 @(Android) Dagger2入门详解 参考文章 环境配置 入门实例 其他注解和情况...
    JFang阅读 594评论 1 0
  • 写在前面:我目前就职于阿里巴巴-菜鸟,团队目前缺人,招聘java和客户端开发,招聘对象为:社招和19届毕业的校招生...
    littleKang阅读 115,379评论 93 745
  • 前言 做这个项目的初衷是想练手,因为现在rxjava+retrofit框架相当火,而公司的同事正在用这个框架也觉得...
    xiaoluYi阅读 2,520评论 15 12
  • MVP https://juejin.im/post/6844903720036073480#heading-14...
    _Anonymous_阅读 251评论 0 0
  • http://zpayh.xyz/2016/07/07/Dagger2%E4%BD%BF%E7%94%A8%E8%...
    奈何心善阅读 191评论 0 0