Dagger2 极速入门

作者: 范蓄能 | 来源:发表于2017-07-21 18:12 被阅读275次

    Github地址

    Dagger是一个快速的依赖注入框架,供Android和Java开发使用。以前由Square维护,现在转交给Google,Github链接为:https://github.com/google/dagger

    依赖注入

    Dagger为依赖注入而生。什么是依赖?什么是注入?为什么要使用依赖注入?这是我们学习Dagger之前必须了解的。依赖就是一个类中要使用其他的类来完成某些工作,这样一个类就依赖了另外一个类。比如在MainActivity中必须使用一个User类的对象,那么MainActivity就依赖了User类。

    public class MainActivity extends AppCompatActivity {
    
        User mUser;//MainActivity依赖User类
    
        @Override
        protected void onCreate(Bundle savedInstanceState) {
            super.onCreate(savedInstanceState);
            mUser = new User();//初始化mUser
            setContentView(R.layout.activity_main);
        }
    }
    

    一般我们直接在MainActivity中直接初始化mUser对象,但如果有天User类的构造方法中增加了一个参数,比如用户名,那么所有调用User的无参构造方法的地方全部要修改,这显然不是我们希望看到的。于是我们思考可不可以提供一个User的工厂类或者容器类专门负责User对象的创建,这样User对象的创建就不会跟MainActivity发生耦合,不管User的构造方法如何变,都不会影响到MainActivity。那么代码可能是这样:

    public class MainActivity extends AppCompatActivity {
    
        User mUser;//MainActivity依赖User类
    
        @Override
        protected void onCreate(Bundle savedInstanceState) {
            super.onCreate(savedInstanceState);
            mUser = UserFactory.getUser()//通过工厂类注入mUser
            setContentView(R.layout.activity_main);
        }
    }
    

    那么mUser对象就不是在MainActivity创建,而是由UserFactory创建,设置给mUser对象,这就可以称之为mUser被注入了,这样MainActivity就不需要关心User对象是怎样创建出来,对User对象创建的修改就不会影响到MainActivity里面的代码,这就是依赖注入的好处的。这就好比我们打针,人体依赖药液来治疗疾病,但药液并不是人体自己生产的,而是医药公司生产然后通过注射器注入到人体,而我们人体根本就无需关心药液是如何被制造出来的。

    Dagger的作用就是通过注解的方式,帮我们自动生成创建对象的工厂类。

    Dagger 2使用

    1. 添加依赖

    可通过链接https://github.com/google/dagger/releases查找最新版本

    dependencies {
        compile 'com.google.dagger:dagger:2.11'
        annotationProcessor 'com.google.dagger:dagger-compiler:2.11'
    }
    

    2. 添加注解@Inject

    分别在MainActivity的User成员变量和User的构造方法上添加@Inject依赖。

    public class MainActivity extends AppCompatActivity {
    
        //成员变量上添加注解
        @Inject
        User mUser;
    }
    
    
    public class User {
    
        String name;
    
        //构造方法上添加注解
        @Inject
        public User() {
            this.name  = "Leon";
        }
    
    }
    

    3. 创建Component类

    事实上完成第二步后,MainActivity中成员mUser并没有调用构造方法完成初始化,还需要一个类来完成注入,这就是Component类。
    我们创建一个MainComponent接口,提供一个inject方法,其参数为将被注入的类MainActivity

    public interface MainComponent {
    
        void inject(MainActivity activity);
    }
    

    然后,我们在Android Studio下选择菜单build->Make Project,这时候,会在app模块下build/generated/source/apt/debug/包名/目录下生成三个文件DaggerMainComponentMainActivity_MembersInjectorUser_Factory。DaggerMainComponent即为接口MainComponent的实现类。

    4. 注入

    最后,我们可以使用生成的DaggerMainComponnet完成注入,成员变量mUser将被赋值。如果是老司机,是不是会觉得这跟ButterKnife.bind(this)有异曲同工之妙呢?

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        //注入,内部会调用User的构造方法完成成员变量mUser的初始化
        DaggerMainComponent.builder().build().inject(this);
        //打印结果,mUser不为null
        Log.d(TAG, "onCreate: " + mUser.name);
    }
    

    Module使用

    假如我想在一个Activity里面注入一个TextView,如下:

    public class MainActivity extends AppCompatActivity {
    
        @Inject
        TextView mTextView;
    }
    

    如果按照上述步骤,就需要找到TextView的构造方法,加上一个@Inject注解,这个显然是无法做到的。Module就是为了解决不能够通过注解构造方法来创建一个实例的问题。我们创建一个TextViewMoudle类,使用@Module注解。另外需要提供一个方法返回一个TextView实例,并且用@Provides注解。

    @Module
    public class TextViewModule {
    
        @Provides
        TextView provideTextView(Context context) {
            return new TextView(context);
        }
    }
    

    provideTextView方法有个Context参数,需要外界传入到TextViewModule,这里给TextViewModule提供一个带有Context参数的构造方法。另外还需要提供个provideContext方法返回上下文,因为Dagger框架调用provideTextView方法获取一个TextView实例时,发现要传一个Context类型的参数,这时候他会查找被@Provides注解并且返回值为Context类型的方法获取一个Context实例传入provideTextView方法。

    @Module
    public class TextViewModule {
        //保存一个上下文成员变量
        private Context mContext;
    
        //构造方法接收一个上下文
        public TextViewModule(Context context) {
            this.mContext = context;
        }
    
        @Provides
        TextView provideTextView(Context context) {
            return new TextView(context);
        }
    
        
        @Provides
        Context provideContext() {
            return mContext;
        }
    }
    

    接下来还需要在MainComponent接口上指定TextViewModule。然后点击菜单选项build->Make Project,这时又会在build/generated/source/apt/debug/包名/目录下生成两个新文件TextViewModule_ProvideContextFactoryTextViewModule_ProvideTextViewFactory

    @Component(modules = TextViewModule.class)
    public interface MainComponent {
    
        void inject(MainActivity activity);
    }
    

    最后在注入时创建一个TextViewModule传入,表示告诉Dagger框架可以从TextViewModule中获取TextView实例。

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        //传入TextViewMoudle
        DaggerMainComponent.builder().textViewModule(new TextViewModule(this)).build().inject(this);
        FrameLayout frameLayout = (FrameLayout) findViewById(R.id.frame);
        //mTextView已被注入,不为null
        mTextView.setText(mUser.name);
        frameLayout.addView(mTextView);
    }

    相关文章

      网友评论

        本文标题:Dagger2 极速入门

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