dagger2从入门到放弃-最基础的用法介绍

前言

dagger2确实学习曲线比较陡峭,但是忘掉生命周期,忘掉局部单例,忘掉SubComponent这样的进阶用法,只用最基础最简单的部分一样可以给项目带来一定程度的便利

依赖的提供方式

  • @Inject注解构造器
  • @Provides注解提供依赖的方法

@Inject注解构造器-用来提供依赖

  • 表示该类可以作为依赖注入到依赖需求方
  • 如果构造器带参数,则这些参数也需要是可以提供的依赖,dagger2会自动找到这些依赖并注入
  • 对于有多个构造器的类,@Inject只能注解其中一个
public class SimpleInjectExtraBean {
    @Inject
    public SimpleInjectExtraBean() {
    }
}

@Inject构造函数带参数的情况

构造函数中的参数需要是@Inject注解构造器的对象或者是@Provides方法提供的依赖

public class SimpleInjectBean extends BaseBean {
    public SimpleModuleBean mSimpleModuleBean;
    public SimpleInjectExtraBean mSimpleInjectExtraBean;

    @Inject
    public SimpleInjectBean(SimpleModuleBean simpleModuleBean,
            SimpleInjectExtraBean simpleInjectExtraBean) {
        mSimpleModuleBean = simpleModuleBean;
        mSimpleInjectExtraBean = simpleInjectExtraBean;
    }
}

@Module @Provides 提供无法修改构造器的依赖

使用@Inject注解构造器是最简单的提供依赖的方式,但是如果是一个第三方库中的代码无法修改构造函数添加注解呢,这时候就需要module出场了

@Module //@Module注解可以提供依赖的类
public class SimpleModule {

    @Provides // @Provides 注解提供依赖的方法
    public SimpleModuleBean provideSimpleModuleBean() {
        return new SimpleModuleBean();
    }
}

@Provides注解的方法带参数

和@Inject注解的构造函数带的参数一样,也需要是@Inject注解构造器的对象或者是@Provides方法提供的依赖

依赖的接收方式

@Inject注解实例变量-用来接受依赖

public class SimpleActivity extends AppCompatActivity {
    @Inject
    SimpleInjectBean mSimpleInjectBean;
    ...
}

连接依赖提供方和依赖接收方的Component

component是依赖提供方和依赖需求方的连接者,使用@Component注解,注意dagger2只能注解接口和抽象类,具体类注解了也生成不了注入相关的代码

对于@Inject不需要在Component中显示指定

对于module,需要在用@Component的modules属性显示指定

@Component(modules = SimpleModule.class)
public interface SimpleComponent {
    void inject(SimpleActivity simpleActivity);
}

框架会生成DaggerXXXComponent的类,使用DaggerXXXComponent完成注入

public class SimpleActivity extends AppCompatActivity {
    @Inject
    SimpleInjectBean mSimpleInjectBean;
    @Inject
    SimpleInjectExtraBean mSimpleInjectExtraBean;
    @Inject
    SimpleModuleBean mSimpleModuleBean;
    ...
    private SimpleComponent mSimpleComponent;

    @Override
    protected void onCreate(@Nullable Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        mSimpleComponent = DaggerSimpleComponent.create();
        mSimpleComponent.inject(this);
        ...
    }
}

以上就用dagger2实现了最简单的依赖注入
不过这基本上接近于写了一个HelloWorld
下面看一下更接近真实的简单场景

提供一个运行时的依赖

dagger2实现依赖注入的实质是创建编译期可以确定的所有类的工厂,在运行时通过dagger2生成的工厂代码(依赖注入框架内部)完成依赖的创建与管理

但是像Application,Activity这样的对象的创建过程是android框架完成的,dagger2无法接管这个过程,而非常多的对象的创建会依赖于context,activity这样的对象

为了解决这样的问题,需要将这些运行时对象作为参数提供给dagger框架

为dagger框架提供参数有两种方式

构造器带参数的Module

以Application为例

@Module
public class BaseAppModule {
    private Application mApplication;

    public BaseAppModule(Application application) {
        mApplication = application;
    }

    @Provides
    @Singleton
    Application provideApplication() {
        return mApplication;
    }

    @Provides
    @Singleton
    Context provideContext() {
        return mApplication;
    }
}

Component的获取方式也会变化

//before Component依赖的所有module的构造函数都没带参数,则可以直接通过DaggerXXXComponent.create()直接获取对应的Component实例
DaggerBaseAppComponent.create();
//等价于DaggerXXXComponent.builder().build();
DaggerBaseAppComponent.builder().build();

//after 如果module中带了参数,则编译期不会生成create()方法
//需要使用.builder()传入对应的Module作为参数,这时可以将Application实例作为Module的构造参数传入
DaggerBaseAppComponent.builder().baseAppModule(new BaseAppModule(this)).build();

直接通过@Component.Builder传入参数

@Component.Builder注解一个用来向Component绑定参数的接口

这里以Activity为例,在Component.Builder中使用@BindsInstance注解一个传入Activity的方法

//provideSimpleModuleActivityBean有一个Activity作为参数
@Module
public class SimpleModule {

    @Provides
    public SimpleModuleActivityBean provideSimpleModuleActivityBean(Activity activity) {
        return new SimpleModuleActivityBean(activity);
    }
}

@Component(modules = SimpleModule.class)
public interface SimpleComponent {
    void inject(SimpleActivity simpleActivity);

    @Component.Builder
    interface Builder {
        //创建Component的时候绑定实例
        @BindsInstance
        Builder simpleActivity(Activity simpleActivity);
        
        SimpleComponent build();
    }
}

注意:如果SimpleModule构造函数还带了参数,因为此时的Component.Builder是按照我们定义的而不是完全自动生成的,所以为能够传入Module中需要的参数,需要显式指定该module作为builder的参数

@Component(modules = SimpleModule.class)
@SimpleScope
public interface SimpleComponent {
    void inject(SimpleActivity simpleActivity);

    @Component.Builder
    interface Builder {
        //创建Component的时候绑定实例
        @BindsInstance
        Builder simpleActivity(Activity simpleActivity);

        //使用了Builder就需要显式的指定带参数的module作为builder的参数
        Builder simpleModule(SimpleModule simpleModule);

        SimpleComponent build();
    }
}

总结

这篇文章介绍了为dagger提供依赖和注入依赖以及提供运行时参数的方法,不过离真实场景还是有一定的区别那就是生命周期的管理

下一篇文章将介绍dagger2中依赖的生命周期管理

相关文章

dagger2从入门到放弃-概念
dagger2从入门到放弃-最基础的用法介绍
dagger2从入门到放弃-Component的继承体系、局部单例
dagger2从入门到放弃-ActivityMultibindings
dagger2从入门到放弃-dagger.android
dagger2从入门到放弃-其他用法
dagger2从入门到放弃-多模块项目下dagger的使用
dagger2从入门到放弃-为何放弃

参考资料

https://google.github.io/dagger/
https://github.com/google/dagger
使用Dagger 2依赖注入
Dagger 2 完全解析

示例代码

DaggerInAction
欢迎star
master分支上最新的代码可能会比当前文章的示例代码稍微复杂点,提交记录里包含了每一步的迭代过程,可以顺藤摸瓜

    原文作者:sunhapper
    原文地址: https://www.jianshu.com/p/71f8bc36c6f5
    本文转自网络文章,转载此文章仅为分享知识,如有侵权,请联系博主进行删除。
点赞