美文网首页Android开发Android开发经验谈Android技术知识
dagger2从入门到放弃-最基础的用法介绍

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

作者: sunhapper | 来源:发表于2018-11-01 20:43 被阅读4次

    前言

    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分支上最新的代码可能会比当前文章的示例代码稍微复杂点,提交记录里包含了每一步的迭代过程,可以顺藤摸瓜

    相关文章

      网友评论

        本文标题:dagger2从入门到放弃-最基础的用法介绍

        本文链接:https://www.haomeiwen.com/subject/twkfxqtx.html