Dependency Injecte(依赖注入)

首先写个不使用依赖注入的示例
  • interface
// House.java
public interface House {
void prepareForWar(); void reportForWar();
}
  • 新建两个实现 House 接口的类
// Starks.java
public class Starks implements House { @Override
public void prepareForWar() {
//do something
System.out.println(this.getClass().getSimpleName()+" prepared for war");
} @Override
public void reportForWar() {
//do something
System.out.println(this.getClass().getSimpleName()+" reporting..");
}
}
// Boltons.java
public class Boltons implements House {
@Override
public void prepareForWar() {
//do something
System.out.println(this.getClass().getSimpleName()+" prepared for war");
} @Override
public void reportForWar() {
//do something
System.out.println(this.getClass().getSimpleName()+" reporting..");
}
}
  • 接着需要一个依赖于这两个的类的类
public class War {

    private Starks starks;

    private Boltons boltons;

    public War(){
starks = new Starks();
boltons = new Boltons(); starks.prepareForWar();
starks.reportForWar();
boltons.prepareForWar();
starks.reportForWar();
} }
  • 下次改用依赖注入的方式实现这个类
public class War {

    private Starks starks;
private Boltons boltons; //DI - getting dependencies from else where via constructor
public War(Starks starks, Boltons bolton){
this.starks = starks;
this.boltons = bolton;
} public void prepare(){
starks.prepareForWar();
boltons.prepareForWar();
} public void report(){
starks.reportForWar();
boltons.reportForWar();
} }
  • 从外部注入依赖的对象
public class BattleOfBastards {

    public static void main(String[] args){

        Starks starks = new Starks();
Boltons boltons = new Boltons(); War war = new War(starks,boltons);
war.prepare();
war.report();
}
}
利用dagger2进行依赖注入
  • Dagger 2 works on Annotation processor. 需要了解一定的java注解知识
  • 首先了解dagger2最常用的2个注解 @Inject@Component

@Inject Annotation

可以作用于

  • 构造器
  • 字段
  • 方法

@Inject注解告诉dagger哪些方法,构造器或者字段是需要依赖注入

But @Inject doesn’t work everywhere:

  • Interfaces can’t be constructed.
  • Third-party classes can’t be annotated.
  • Configurable objects must be configured!

@Component Annotation

作用于接口

@Componentdagger会生成一个实现该接口的class,该class会实现其中的方法,提供需要依赖的对象,相当于是一个代理

修改的代码

添加默认构造器,并添加注解@Inject

public class Boltons implements House {

    // 添加默认构造器,并添加注解@Inject
@Inject
public Boltons(){
} @Override
public void prepareForWar() {
//do something
System.out.println(this.getClass().getSimpleName()+" prepared for war");
} @Override
public void reportForWar() {
//do something
System.out.println(this.getClass().getSimpleName()+" reporting..");
}
}
public class Starks implements House {

    @Inject //Dagger 2
public Starks(){
} @Override
public void prepareForWar() {
//do something
System.out.println(this.getClass().getSimpleName()+" prepared for war");
} @Override
public void reportForWar() {
//do something
System.out.println(this.getClass().getSimpleName()+" reporting..");
}
}
public class War {

    private Starks starks;

    private Boltons boltons;

    @Inject
public War(Starks starks, Boltons bolton){
this.starks = starks;
this.boltons = bolton;
} public void prepare(){
starks.prepareForWar();
boltons.prepareForWar();
} public void report(){
starks.reportForWar();
boltons.reportForWar();
} }
  • 修改依赖加载方式
public class BattleOfBastards {

    public static void main(String[] args){
// Mannual DI
// Starks starks = new Starks();
// Boltons boltons = new Boltons();
// War war = new War(starks,boltons);
// war.prepare();
// war.report(); // Using Dagger 2
BattleComponent component = DaggerBattleComponent.create();
War war = component.getWar();
war.prepare();
war.report(); }
}
在build之后查看annotation processor 自动生成的代码
  • 如图

  • DaggerBattleComponent.java 为dagger自动生成的代码,已类名加上Dagger前缀命名, 该类实现了BattleComponent接口
package com.explore.lin.didemo.javaDIdemo;

import javax.inject.Provider;

public final class DaggerBattleComponent implements BattleComponent {
private Provider<War> warProvider; private DaggerBattleComponent(Builder builder) {
assert builder != null;
initialize(builder);
} public static Builder builder() {
return new Builder();
} public static BattleComponent create() {
return new Builder().build();
} @SuppressWarnings("unchecked")
private void initialize(final Builder builder) { this.warProvider = War_Factory.create(Starks_Factory.create(), Boltons_Factory.create());
} @Override
public War getWar() {
return new War(new Starks(), new Boltons());
} public static final class Builder {
private Builder() {} public BattleComponent build() {
return new DaggerBattleComponent(this);
}
}
}
  • 继续在component中添加方法
@Component
interface BattleComponent {
War getWar();
//adding more methods
Starks getStarks();
Boltons getBoltons();
}
  • 观察dagger2生成的DaggerBattleComponent中
  // DaggerBattleComponent.java
@Override
public War getWar() {
return new War(new Starks(), new Boltons());
} @Override
public Starks getStarks() {
return new Starks();
}
  • 如果我们删除Boltons.java构造器中的注解@Inject,会发现无法通过编译,因为在war.java的构造器中存在对Boltons的依赖
总结

比如有个classA中存在对classB的依赖,用dagger2怎么实现呢

  • 一种
class A {

    @Inject
public A() {
} @Inject
B b; void act() {
b.prepare();
}
} class B {
@Inject
public B() { } void prepare() {
System.out.println("b.prepare()");
}
} @Component
interface AComponent {
A a();
} public class InjectDemo { public static void main(String[] args) {
DaggerAComponent.create().a().act();
}
}
  • 二,什么时候使用@Provide,比如你使用第三方库,或则B中构造器没有@Inject
class AP {
@Inject
public AP() { }
@Inject
BP bp; void act() {
bp.prepare();
}
} class BP {
void prepare() {
System.out.println("bp.prepare()");
}
} @Component(modules = {APModule.class})
interface APComponent{
AP ap();
} @Module
class APModule { @Provides
BP providerBP() {
return new BP();
}
} public class ProvideDemo {
public static void main(String[] main) {
DaggerAPComponent.create().ap().act();
}
}
在android中应用dagger
  • 同样先新建不使用dagger的项目

参考https://github.com/DaiHangLin/dependencyInjecte

android dagger2使用笔记的更多相关文章

  1. Android自动化学习笔记:编写MonkeyRunner脚本的几种方式

    ---------------------------------------------------------------------------------------------------- ...

  2. Android自动化学习笔记之MonkeyRunner:官方介绍和简单实例

    ---------------------------------------------------------------------------------------------------- ...

  3. android开发学习笔记000

    使用书籍:<疯狂android讲义>——李刚著,2011年7月出版 虽然现在已2014,可我挑来跳去,还是以这本书开始我的android之旅吧. “疯狂源自梦想,技术成就辉煌.” 让我这个 ...

  4. Android动画学习笔记-Android Animation

    Android动画学习笔记-Android Animation   3.0以前,android支持两种动画模式,tween animation,frame animation,在android3.0中 ...

  5. Android 数字签名学习笔记

    Android 数字签名学习笔记 在Android系统中,所有安装到系统的应用程序都必有一个数字证书,此数字证书用于标识应用程序的作者和在应用程序之间建立信任关系,如果一个permission的pro ...

  6. Android高级编程笔记(四)深入探讨Activity(转)

    在应用程序中至少包含一个用来处理应用程序的主UI功能的主界面屏幕.这个主界面一般由多个Fragment组成,并由一组次要Activity支持.要在屏幕之间切换,就必须要启动一个新的Activity.一 ...

  7. Android群英传笔记——第十二章:Android5.X 新特性详解,Material Design UI的新体验

    Android群英传笔记--第十二章:Android5.X 新特性详解,Material Design UI的新体验 第十一章为什么不写,因为我很早之前就已经写过了,有需要的可以去看 Android高 ...

  8. Android群英传笔记——第十章:Android性能优化

    Android群英传笔记--第十章:Android性能优化 随着Android应用增多,功能越来越复杂,布局也越来越丰富了,而这些也成为了阻碍一个应用流畅运行,因此,对复杂的功能进行性能优化是创造高质 ...

  9. Android群英传笔记——第九章:Android系统信息和安全机制

    Android群英传笔记--第九章:Android系统信息和安全机制 本书也正式的进入尾声了,在android的世界了,不同的软件,硬件信息就像一个国家的经济水平,军事水平,不同的配置参数,代表着一个 ...

随机推荐

  1. MySQL和PostgreSQL比较

    1.MySQL相对来说比较年轻,首度出现在1994年.它声称自己是最流行的开源数据库.MySQL就是LAMP(用于Web开发的软件包,包括 Linux.Apache及Perl/PHP/Python)中 ...

  2. HDU 4005 The war 双连通分量 缩点

    题意: 有一个边带权的无向图,敌人可以任意在图中加一条边,然后你可以选择删除任意一条边使得图不连通,费用为被删除的边的权值. 求敌人在最优的情况下,使图不连通的最小费用. 分析: 首先求出边双连通分量 ...

  3. 做一个日收入100元的APP!

    [导语]虽然讲了很多个人开发者的文章,但新手开发者如何赚自己的第一个100块钱,确是最难的事情.群里有人说都不知道干什么 app赚钱,完全没有想法, 并且经常问我有什么快速赚钱的方法.我只能遗憾地说, ...

  4. WINDOWS批量替换不同文件夹下的相同文件

    今天帮媳妇解决的问题,记录一下,也许以后有用 例子: N个文件夹下有同一个文件(common.php),但是,现在对common.php文件进行了大量修改. 现在想用最新的common.php替换掉所 ...

  5. 获取表的字段例如 col1,col2,col3

    create function [dbo].[f_getcolsByName](@tableName varchar(50)) returns varchar(1000)asbegin declare ...

  6. maven学习(三)——修改maven本地默认仓库

    修改从Maven中心仓库下载到本地的jar包的默认存储位置 从Maven中心仓库下载到本地的jar包的默认存放在”${user.home}/.m2/repository”中,${user.home}表 ...

  7. CentOS 6.4下编译安装MySQL 5.6.14 (转)

    CentOS 6.4下通过yum安装的MySQL是5.1版的,比较老,所以就想通过源代码安装高版本的5.6.14. 正文: 一:卸载旧版本 使用下面的命令检查是否安装有MySQL Server rpm ...

  8. ansible中playbook使用

    palybook使用 ####yaml语法ansible中使用的yaml基础元素:变量Inventory条件测试迭代 playbook的组成结构InventoryModulesAd Hoc Comma ...

  9. 通过sql查询rman备份信息

    通过sql查询rman备份信息 查看所有备份集 SELECT A.RECID "BACKUP SET", A.SET_STAMP, DECODE (B.INCREMENTAL_LE ...

  10. 一道背包神题-Petrozavodsk Winter-2018. Carnegie Mellon U Contest Problem I

    题目描述 有\(n\)个物品,每个物品有一个体积\(v_i\),背包容量\(s\).要求选一些物品恰好装满背包且物品个数最少,并在这样的方案中: (1)求出中位数最小的方案的中位数(\(k\)个元素的 ...