听说你又一次放弃了Dagger2,来听老夫说道说道
关于Dagger2是啥,用法网上已经有很多了。不过听说你已经很多次Dagger2从入门到放弃了,来来,让我这个老中医给你治一治。老夫手法纯熟,不会让你有任何痛苦的,放心读下去吧。
本文同步自 博主的私人博客 wing的地方酒馆
诊断
咳咳,看你这架势病的不清,应该是陌生代码恐惧症,你看看下面的代码。会不会害怕。
private ShipDetailPresenter mShipDetailPresenter;
mShipDetailPresenter = DaggerShipDetailComponent.builder()
.shipDetailModule(new ShipDetailModule(this, this))
.build()
.inject();
喂!!!你怎么跑了,快回来,还没治病呢!!
用法回顾(注入第一式)
首先,对于一般来说,想创建一个对象,我们肯定采取new的形式:
mShipDetailPresenter = new ShipDetailPresenter(context,shipDetailView);
但是这样耦合度较高,presenter的变化受限于Activity,也就是说假如我们的presenter的参数变成三个了,又要改Activity代码。所以使用Dagger2的好处就很明显了,如果改了构造器,只需要改对应的Module就可以。
@Module
上面提到了Module,这是什么呢?其实这是一个提供参数的地方,将new Presenter所需要的参数,写到这里面来了。看示例代码:
@Module //注解说明这是Module
public class ShipDetailModule {
private ShipDeatilContract.View mView;
private Context mContext;
public ShipDetailModule(ShipDeatilContract.View mView, Context mContext) {
this.mView = mView;
this.mContext = mContext;
}
//注解说明Provide View 既提供View参数
@Provides ShipDeatilContract.View getView(){
return mView;
}
//注解说明Provide Context 既提供Context参数
@Provides Context getContext(){
return mContext;
}
}
这是Module的写法,就是把new Presenter(Context,View)写到了里面。
@Inject
Module怎么知道presenter构造器需要几个参数呢,答案是@Inject ,只需要在Presenter的构造器注解@Inject
public class ShipDetailPresenter implements ShipDeatilContract.Presenter {
Context context;
ShipDeatilContract.View shipDetailViewInterface;
//告知构造求所需参数
@Inject
public ShipDetailPresenter(Context context,ShipDeatilContract.View shipDetailViewInterface){
this.context=context;
this.shipDetailViewInterface=shipDetailViewInterface;
}
@Component
连接Presenter和Module的东东,我们只需要写一个接口,来返回Presenter对象,Dagger2就会自动生成对应的对象给我们。
//注解标注Component 并且告知需要的module是什么
@Component(modules = ShipDetailModule.class)
public interface ShipDetailComponent {
//接口的方法,返回presenter
ShipDetailPresenter inject();
}
最终联系
现在看回第一段代码
private ShipDetailPresenter mShipDetailPresenter;
mShipDetailPresenter = DaggerShipDetailComponent.builder()
.shipDetailModule(new ShipDetailModule(this, this))
.build()
.inject();
DaggerShipDetailComponent这个类是build之后Dagger2自动为我们生成的类,查看源码它实现了ShipDetialComponent这个方法,所以inject()函数自然会为我们返回一个presenter来啦。
用法回顾(注入第二式)
你可能犯嘀咕了,你这种写法不科学!跟我看到的不一样。咳咳,不要急,老夫再给你说另一种用法。这种方法的代码如下:
@Inject ShipDetailPresenter mShipDetailPresenter;
DaggerShipDetailComponent.builder()
.shipDetailModule(new ShipDetailModule(this, this))
.build()
.inject(this);
对应的Component为:
public interface ShipDetailComponent {
void inject(ShipDetailActivity);
}
其他的写法都一样。 此时你心里肯定犯嘀咕了,么么的,你刚才说是inject()返回presenter所以可以赋值,这个void你怎么解释! 是不是又要放弃拉,年轻人,不要轻言放弃,听老夫一言。
想要知道Dagger2 怎么做的那么神奇,必须知道他的原理,原理怎么看?当然是看他生成的代码呗!
Dagger2原理
第一式解密
首先看第一种方法的原理,找到DaggerShipDetailComponent 发现代码如下:
public final class DaggerShipDetailComponent implements ShipDetailComponent {
private Provider<Context> getContextProvider;
private Provider<ShipDeatilContract.View> getViewProvider;
private Provider<ShipDetailPresenter> shipDetailPresenterProvider;
/** 去处多行代码**/
}
可以看到Component确实实现了我们定义的Component,所以他的inject()方法肯定是返回了一个presenter
@Override
public ShipDetailPresenter inject() {
return shipDetailPresenterProvider.get();
}
老夫没有骗你吧,至于这个provider如何拿到的newPresenter,这里给你留个家庭作业。现在应该有些头绪了吧,Dagger2没有想象中那么难,对不对?接下来看看让你懵逼的第二式。
第二式
直接看生成的代码!在DaggerShipDetailComponent发现了
public void inject(ShipDetailActivity activity) {
shipDetailsMembersInjector.injectMembers(activity);
}
继续往下看,看到了一个init方法
private void initialize(final Builder builder) {
this.getContextProvider = ShipDetailModule_GetContextFactory.create(builder.shipDetailModule);
this.getViewProvider = ShipDetailModule_GetViewFactory.create(builder.shipDetailModule);
//工厂生成provider 用来提供presenter
this.shipDetailPresenterProvider =
ShipDetailPresenter_Factory.create(getContextProvider, getViewProvider);
//生成MembersInjector
this.shipDetailActivityMembersInjector =
ShipDetailActivity_MembersInjector.create(shipDetailPresenterProvider);
}
接下来顺藤摸瓜,找到了ShipDetailActivity_MembersInjector,只看关键部分源码
public final class ShipDetailActivity_MembersInjector
implements MembersInjector<ShipDetailActivity> {
//进行注入操作
public static void injectShipDetailPresenter(
ShipDetailActivity instance, Provider<ShipDetailPresenter> shipDetailPresenterProvider) {
instance.shipDetailPresenter = shipDetailPresenterProvider.get();
}
}
相信聪明的你已经明白了,为什么返回void还能注入,不知道你有没有发现,第二式的presenter没有用private修饰,在injectShipDetailPresenter(ShipDetailActivity,Provider)的时候将 activity.presenter = Proivider.get()!
如果让我用两个字来形容的话,我只能说
太他妈精彩
嘛,这次会诊就到这里吧,回家记得勤加锻炼,早日康复,再见。
听说你又一次放弃了Dagger2,来听老夫说道说道的更多相关文章
- 掘金 Android 文章精选合集
掘金 Android 文章精选合集 掘金官方 关注 2017.07.10 16:42* 字数 175276 阅读 50053评论 13喜欢 669 用两张图告诉你,为什么你的 App 会卡顿? - A ...
- ajax异步上传到又拍云的实例教程
作者:白狼 出处:www.manks.top/article/async_upload_to_upyun 本文版权归作者,欢迎转载,但未经作者同意必须保留此段声明,且在文章页面明显位置给出原文连接,否 ...
- [转载] TLS协议分析 与 现代加密通信协议设计
https://blog.helong.info/blog/2015/09/06/tls-protocol-analysis-and-crypto-protocol-design/?from=time ...
- TLS协议分析
TLS协议分析 本文目标: 学习鉴赏TLS协议的设计,透彻理解原理和重点细节 跟进一下密码学应用领域的历史和进展 整理现代加密通信协议设计的一般思路 本文有门槛,读者需要对现代密码学有清晰而系统的理解 ...
- 关于Dagger 2的文章汇总
首先是我真正看懂了的第一篇文章 代码GG之家的 Dagger2图文详解 这篇文章很直接,还配有代码demo.至少我是看懂了. Dagger2 使用详解 这篇文章同样配有demo,同时文末还有很多有用的 ...
- PayPal高级工程总监:读完这100篇论文 就能成大数据高手(附论文下载)
100 open source Big Data architecture papers for data professionals. 读完这100篇论文 就能成大数据高手 作者 白宁超 2016年 ...
- Asp.net 面向接口可扩展框架之核心容器(含测试代码下载)
新框架的容器部分终于调通了!容器实在太重要了,所以有用了一个名词叫“核心容器”. 容器为什么那么重要呢?这个有必要好好说道说道. 1.首先我们从框架名称面向接口编程说起,什么是面向接口编程?(这个度娘 ...
- 算法系列:FFT 001
转载自http://blog.csdn.net/orbit/article/details/17210461 2012年9月的时候,一个南京的大学生从电视台播放的一段记者采访360总裁周鸿祎的视频中破 ...
- redis入门笔记(2)
redis入门笔记(2) 上篇文章介绍了redis的基本情况和支持的数据类型,本篇文章将介绍redis持久化.主从复制.简单的事务支持及发布订阅功能. 持久化 •redis是一个支持持久化的内存数据库 ...
随机推荐
- C++函数的重载
两个以上的函数,具有相同的函数名,但是形参的个数或者类型不同,编译器根据实参与形参的类型以及个数的最佳匹配,自动确定调用的函数,这就是函数的重载. 两个名字相同的函数必须具有不同的形参,这里的不同指的 ...
- 整理spring定时器corn表达式
1.结构 corn从左到右(用空格隔开):秒 分 小时 月份中的日期 月份 星期中的日期 年份 2.各字段的含义 字段 允许值 允许的特殊字符 秒 0~59 - * / 分 0~59 - * / ...
- javascript实现图片的预览
简单javascript代码 实现上传图片预览 <body> <!-- 设置当有图片准备上传时触发javascript代码--> <input type="fi ...
- Jenkins执行批处理文件、powershell失败
今天搭建Jenkins持续集成环境,编译环境是.net core.整理了一些发布的命令配置在Jenkins,问题来了,使用powershell插件运行dotnet restore.dotnet bui ...
- CentOS Linux上安装Oracle11g笔记
CentOS Linux上安装Oracle11g 到 otn.oracle.com 网站上下载 Linux版的oracle 11g 编辑 /etc/sysctl.conf : kernel.shmal ...
- SpringSecurity 进行自定义Token校验
背景 Spring Security默认使用「用户名/密码」的方式进行登陆校验,并通过cookie的方式存留登陆信息.在一些定制化场景,比如希望单独使用token串进行部分页面的访问权限控制时,默认方 ...
- Angular4.0入门
angular与其他的差别 angular cli安装 cnpm install -g @angular/cli 最新版本 cnpm uninstall -g @angular/cli 卸载全局版本 ...
- JVM三种垃圾收集算法思想及发展过程
JVM垃圾收集算法的具体实现有很多种,本文只是介绍实现这些垃圾收集算法的三种思想和发展过程.所有的垃圾收集算法的具体实现都是遵循这三种算法思想而实现的. 1.标记-清除算法 标记-清除(Mark-Sw ...
- Android 实现串口的移植
安卓串口的实现,需要底层C++配合,不过这次我们根据framework中的思想,直接用API修改提供给JAVA层调用,这个就比较简单了. DEV项目需要,要实现在Android中实现串口的收发功能,有 ...
- Mac 下安装运行Rocket.chat
最近花了一周的时间,复习了HTML.CSS.原生JS,并学习了Node.js.CoffeeScript.js.MongoDB,入了下门. 因为准备在Rocket.chat 上做二次开发,所以先下载和安 ...