dagger2 学习(一)

前言

该部分代码

记录一下自己的基本的认识,Dagger 2 的目的是为了解决依赖问题

当我们申明一个 User mUser 对象 ,此时mUser 并未实例化,而当我们需要实例化的时候,则需要调用new User()

而使用Dagger 2 的时候,我们只需要 @inject User mUser 然后在Activity中调用DaggerXXXX....inject(this)即可(假设是Activity,也可以是其他任何类),如下

@User mUser;
@ClassRoom mClassRoom;
...

DaggerXXXX....inject(this)// 只需要调用一次即可
...
mUser.get...
mClassRomm.get...

那为什么不直接new User() 呢,因为会面对一个依赖问题,而下面的例子并没有突出解决这个问题,所以暂且搁置不写,讨论到 @Module 时继续讨论( 面对 new User(params...) 时,需要其他依赖时, 配合 @Module 这个注解进行使用,会显得更干净)。

下面的记录就是最简单的注入,主要搞定了 new User() 这类无依赖的注入问题

Dagger 2 基本使用

1.@inject 使用, 一共两种使用地方

(1) 标记被注入的对象的构造方法
public class User {
    int age;
    @Inject// 标注构造方法
    public User(){
    }
  
    public int getAge() {
        return age;
    }

    public void setAge(int age) {
        this.age = age;
    }
}
(2) 标记注入目标处
public class MainActivity extends AppCompatActivity {


    TextView mUserAgeTv;
    @Inject User mUser;// 标注注入的对象


    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        mUserAgeTv = (TextView) findViewById(R.id.user_age_tv);
    }
}

这样标注出了两个地方,一个被注入对象的构造方法,一个需要注入的对象,但是相互之间是没有关联的

写完这两个类,编译后,看一下生成的具体代码,验证一下

(3) @Inject public User()... 后生成的代码
public enum User_Factory implements Factory<User> {
  INSTANCE;

  @Override
  public User get() {
    return new User();// 真正生成 User 对象的地方
  }

  public static Factory<User> create() {
    return INSTANCE;// 返还User_Factory 对象
  }
}
(4) @Inject User mUser 生成的代码
public final class MainActivity_MembersInjector implements MembersInjector<MainActivity> {
  private final Provider<User> mUserProvider;// 用来提供User对象的 Provider

  public MainActivity_MembersInjector(Provider<User> mUserProvider) {
    assert mUserProvider != null;
    this.mUserProvider = mUserProvider;
  }

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

  @Override
  public void injectMembers(MainActivity instance) {
    if (instance == null) {
      throw new NullPointerException("Cannot inject members into a null reference");
    }
    // 看起来像注入的地方 调用了 userProvider 然后调用了 get
    instance.mUser = mUserProvider.get();
  }

  public static void injectMUser(MainActivity instance, Provider<User> mUserProvider) {
    // 这个也是看起来像注入的地方 调用了 userProvider 然后调用了 get
    instance.mUser = mUserProvider.get();
  }
}

仔细看,发现两者生成的代码现在是无关的,接下来引入连接的注解

2.@Component 使用,连接上述的两个地方,进行注入

(1)使用

加入一个新的类,如下

@Component // 注解标记
public interface UserComponent {
    void injectTo(MainActivity mainActivity);
  // 注意此处方法的参数 MainActivity ,由此标记处要注入的类,而方法名其实可以是任何名字
  // 被 @Component 标注的类,其具体编译后生成的代码,会有具体的实现类来处理,参考下文
}

看文档,只知道这是一个连接的,也不知道具体怎么连接的,只知道会生成一个DaggerUserComponent 类,生成代码在具体调用分析处,可具体参考

MainActivity加入 DaggerUserComponent.builder().build().injectTo(this) 就可以注入成功,如下:

MainActivity.java
  

public class MainActivity extends AppCompatActivity {
    TextView mUserAgeTv;
    @Inject User mUser;
    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        mUserAgeTv = (TextView) findViewById(R.id.user_age_tv);// 加入这句
        DaggerUserComponent.builder().build().injectTo(this);
        mUserAgeTv.setText("" + mUser.getAge());
    }
}
(2)那就结合调用整体,具体看下生成的代码吧

DaggerUserComponent.builder().build().injectTo(this);

调用主要分两条线,看注释

  • ..builder().build()

    1 -> 2 步骤 生成一个 DaggerUserComponent 对象,同时实例化出 mainActivityMembersInjector

  • ..injectTo(this);

    3 步骤 是调用 MainActivity_MembersInjector 进行真正的注入

DaggerUserComponent.java
  

public final class DaggerUserComponent implements UserComponent {
  private MembersInjector<MainActivity> mainActivityMembersInjector;

  private DaggerUserComponent(Builder builder) {
    assert builder != null;
    initialize(builder);//2.1 初始化 builder 对象
  }

  public static Builder builder() {
    return new Builder();// 1. 首先生成一个 Builder 对象
  }

  public static UserComponent create() {
    return builder().build();
  }

  @SuppressWarnings("unchecked")
  private void initialize(final Builder builder) {
    // 2.2 关键处,它调用的是之前生成 MainActivity_MembersInjector 的 create 方法
    this.mainActivityMembersInjector = MainActivity_MembersInjector.create(User_Factory.create());
  }

  @Override
  public void injectTo(MainActivity mainActivity) {//3. 最后调用 injectTo 方法
    // 3.1 mainActivityMembersInjector 它由 2.2 转换出来
    mainActivityMembersInjector.injectMembers(mainActivity);
  }

  public static final class Builder {
    private Builder() {}

    public UserComponent build() {
      return new DaggerUserComponent(this);//2. 生成一个 DaggerUserComponent 对象
    }
  }
}

综上,整个连接过程就出来了

下图中 黄色部分,就是builder().build() 构建出MainActivity_MembersInjector,中间调用了 User_Factory 来生成mUserProvider

下图紫色部分,就是injectTo() 进行注入,根据构建出的MainActivity_MembersInjector,获取 User 构建的实例,进行赋值

dagger2学习(一)

最后

整个记录主要写了 Dagger 2 最基本的使用,处理了 new User() 此类操作。

当然这不是 Dagger 2 的精华部分,此处主要是为了看其生成的代码,及其基本的使用。

其生成的对应关系如下:

@User mUser ===> User_Factory.java

@User public User(){} ===> MainActivity_MembersInjector.java (其中的 Provider<User> mUserProvider )

@Component UserComponent{...} ===> DaggerUserComponent.java

其具体作用在上面的代码注释分析中已经写了,此处不重复

以上。

一家之言,如有错误,轻喷。

该部分代码

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

推荐阅读更多精彩内容

  • Dagger2 转载请注明原作者,如果你觉得这篇文章对你有帮助或启发,可以关注打赏。 前言本文翻译自Google ...
    轻云时解被占用了阅读 6,566评论 4 31
  • Spring Cloud为开发人员提供了快速构建分布式系统中一些常见模式的工具(例如配置管理,服务发现,断路器,智...
    卡卡罗2017阅读 134,049评论 18 139
  • 本文更新于2017年11月15日阅读本文大概需要三分半。 前言 dagger出来已经很长时间了,技术的更新日新月异...
    月下溪明阅读 2,825评论 1 9
  • 当曙光爬上黎明的窗台,绯红的云霞旖旎了窗外的天空,我子弹般弹起,破门而出,一头扎进冷嗖嗖的校园,急切地作别连日的阴...
    慕容兰馨阅读 283评论 0 4
  • 前些天,学校里楼管大叔拿着智能手机让我帮忙搜一部《沙家浜》。折腾了一会,连上了网络并教给他。事后我不禁感叹这是为什...
    时间与区间阅读 236评论 0 0