安卓基础开发库,让开发简单点。
DevRing & Demo地址:https://github.com/LJYcoder/DevRing
学习/参考地址:
https://www.jianshu.com/p/cd2c1c9f68d4
https://blog.csdn.net/lisdye2/article/details/51942511
https://www.jianshu.com/p/24af4c102f62
Dagger2已经出来挺久了,网上相关的教程也很多,但我还是坚持要写这篇,做事要有始有终嘛(这应该是本系列介绍的最后一个框架了,也是本系列最难上手的一个框架)
由于接触Dagger2的时间不长,有些理解可能也不到位,但我还是会尽我所能把它讲清楚些,不当之处也请大家指出。
Dagger2是一个依赖注入(Dependency Injection)框架。
什么又是依赖注入呢?
借别人的话来说,就是“目标类中所依赖的其他类的初始化过程,不是通过在目标类中编码的方式完成,而是通过其他手段把已经初始化好的实例自动注入到目标类中”。
再换种方式来说,就是把类实例的初始化过程都挪至一个地方统一管理,具体需要哪个实例时,就从这个地方取出(注入到目标类中)。
知其然,然后要知其所以然。
假设有一个A类,项目中很多地方都使用到它(在很多地方通过new A()对A实例进行了初始化)。然后由于需求变动,A的构造函数增加了一个参数。
好了,牵一发而动全身,你需要把各个new A()的地方都进行修改。
但如果是使用Dagger2进行管理,你只需在类实例的供应端进行修改即可。
假设现在你需要调用A类的x()方法来实现某功能,但是A类的构造过程相当的复杂(这样的例子可以参考GreenDao中获取XXXDao、Retrofit中获取Observable请求)
public void xxx(){
E e = new E();
D d = new D(e);
C c = new C();
B b = new B(c,d);
A a = new A(b);
a.x();
}
结果6行代码中,构造实例a占了5行,调用x()方法实现功能却只占了1行。
但如果使用Dagger2进行管理,将a实例的构造过程移至实例供应端,则功能实现模块的代码会变成这样
@Injcet
A a;
public void xxx(){
a.x();
}
这就是所说的让功能实现更专注于功能实现,而不必去管a实例的构造过程。
通常我们开发中会有两种类实例:
一种是全局实例(单例),它们的生命周期与app保持一致。
一种是页面实例,它们的生命周期与页面保持一致。
通过Dagger2,我们可以使用一个组件专门管理全局类实例(也免去了单例的写法,不用考虑饿汉懒汉什么的);然后各个页面也有各自组件去管理它们的页面实例。
这样不管是对于实例的管理,还是项目的结构,都会变得更加的清晰明了。
(这点可以略过…)
当你不认识Dagger2却看着使用Dagger2的项目代码,很可能会感觉到一种茫然与高大上:
各种@Inject,@Provides,@Singleton,Lazy<>,Provider<>等是什么鬼?
为什么没有实例化的代码?
为什么明明是null却不会报空指针?
等你学会使用Dagger2之后,你也可以来一波”高逼格”的代码。
当然,这里也不得不提一下,使用Dagger2会增加代码量。所以如果是小项目/独立开发,你也可以考虑不用,因为你可能有种失大于得的感觉。如果是大项目/团队开发,使用后就得大于失了。
在讲用法前,先对几个重要角色进行了解。
一个类中,它包含了实例成员,在使用这些实例前,需要对它们进行初始化,但前面已经说了初始化的过程挪至其他地方。所以这个类的需求是,已经完成初始化的实例对象。
暂且把这样的类称为“实例需求端”。
进行实例初始化的地方,并对外供应所需的实例。
Dagger2中,供应端有两种方式可以供应实例:(后面会介绍)
第一种:使用Module
第二种:使用@Inject标注构造函数
实例供应端并不知道它要供应的实例交给谁。
这个时候就需要通过桥梁,将实例需求端与实例供应端联系在一起。
Dagger2中,Component就负责扮演桥梁的角色。
compile 'com.google.dagger:dagger:2.14.1'
annotationProcessor 'com.google.dagger:dagger-compiler:2.14.1'
在实例需求端中,使用@Inject标注需要注入的实例变量。
public class UploadActivity extends AppCompatActivity{
@Inject
UploadPresenter mPresenter;
}
前面说了,供应端有两种方式可以供应实例。
@Module
public class UploadActivityModule {
@Provides
UploadPresenter uploadPresenter() {
return new UploadPresenter();
}
}
public class UploadPresenter{
@Inject
public UploadPresenter() {
}
}
方式一的优先级高于方式二,意思就是:
- 在供应某实例时,会先通过方式一查找是否存在返回该实例的的方法
- 如果存在,则获取实例并对外供应
- 如果不存在,再通过方式二查找是否存在@Inject标注的构造函数
- 如果存在,则将通过该构造函数构造实例并对外供应
- 如果不存在,那将报错,因为无法供应所需的实例
@Component(modules = UploadActivityModule.class)
public interface UploadActivityComponent {
void inject(UploadActivity uploadActivity);
}
public class UploadActivity extends AppCompatActivity{
@Inject
UploadPresenter mPresenter;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
//注入实例
DaggerUploadActivityComponent.builder()
.build()
.inject(this);
//注入后即可调用mPresenter中的方法了
mPresenter.xxx();
}
}
有些时候,实例的初始化需要接收外部参数才能完成,比如MVP中的Presenter往往需要传入IView接口以便完成数据回调。
现在UploadPresenter的构造函数发生了变动,需传入IUploadView。
public class UploadPresenter{
IUploadView mIView;
public UploadPresenter(IUploadView iview) {
mIView = iview;
}
}
下面介绍两种方式来接收外部参数
@Module
public class UploadActivityModule {
IUploadView mIView;
public UploadActivityModule(IUploadView iview) {
mIView = iview;
}
@Provides
IUploadView iUploadView(){
return mIView;
}
@Provides
UploadPresenter uploadPresenter(IUploadView iview) {
return new UploadPresenter(iview);
}
}
public class UploadActivity implements IUploadView{
@Inject
UploadPresenter mPresenter;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
//注入实例
DaggerUploadActivityComponent.builder()
.uploadActivityModule(new UploadActivityModule(this))//通过构造函数传入外部参数IUploadView
.build()
.inject(this);
//注入后即可调用mPresenter中的方法了
mPresenter.xxx();
}
//实现IUploadView接口的方法
@Override
public void onUploadSuccess() {
//上传成功
}
@Override
public void onUploadFail() {
//上传失败
}
}
@Component(modules = UploadActivityModule.class)
public interface UploadActivityComponent {
void inject(UploadActivity uploadActivity);
IUploadView iUploadView();
@Component.Builder
interface Builder {
@BindsInstance
Builder iUploadView(IUploadView iUploadView);
UploadActivityComponent build();
}
}
public class UploadActivity extends AppCompatActivity implements IUploadView{
@Inject
UploadPresenter mPresenter;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
//注入实例
DaggerUploadActivityComponent.builder()
.iUploadView(this)
.build()
.inject(this);
//注入后即可调用mPresenter中的方法了
mPresenter.xxx();
}
//实现IUploadView接口的方法
@Override
public void onUploadSuccess() {
//上传成功
}
@Override
public void onUploadFail() {
//上传失败
}
}
@Qualifier主要是用于解决,因供应端存在多个类型相同的实例而引起歧义的问题。
Dagger2默认提供了一个@Named注解,从代码可以看出属于@Qualifier的一种实现。
@Qualifier
@Documented
@Retention(RUNTIME)
public @interface Named {
String value() default "";
}
下面举个例子,现在页面需有两个Dialog,一个用于登录,一个用于注册。
@Module
public class TestActivityModule {
private Context mContext;
public TestActivityModule(Context context){
mContext = context;
}
@Provides
Context context(){
return mContext;
}
@Named("login")
@Provides
Dialog loginDialog(Context context){
Dialog dialog = new Dialog(context);
dialog.setTitle("登录提示");
....
return dialog;
}
@Named("register")
@Provides
Dialog registerDialog(Context context){
Dialog dialog = new Dialog(context);
dialog.setTitle("注册提示");
....
return dialog;
}
}
@Component(modules = TestActivityModule.class)
public interface TestActivityComponent {
void inject(TestActivity testActivity);
}
public class TestActivity extends AppCompatActivity{
@Named("login")
@Inject
Dialog mDialogLogin;
@Named("register")
@Inject
Dialog mDialogRegister;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
//注入实例
DaggerTestActivityComponent.builder()
.testActivityModule(new TestActivityModule(this))
.build()
.inject(this);
}
}
使用@Named注解的话,需要加入字符串来区分,这样比较麻烦也容易出错。所以我们可以使用自定义的限定符注解。
@Qualifier
public @interface DialogLogin {
}
@Qualifier
public @interface DialogRegister {
}
然后把前面涉及的@Named(“login”)换成@DialogLogin,@Named(“register”)换成@DialogRegister即可~
@Scope的作用是使同一个Component中供应的实例保持唯一。
举例说明:
public class UploadActivity extends AppCompatActivity{
@Inject
UploadPresenter mPresenter1;
@Inject
UploadPresenter mPresenter2;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
//注入实例
DaggerUploadActivityComponent.builder()
.build()
.inject(this);
}
}
@Scope
@Documented
@Retention(RUNTIME)
public @interface ActivityScope {}
@ActivityScope
@Component(modules = UploadActivityModule.class)
public interface UploadActivityComponent {
void inject(UploadActivity uploadActivity);
}
@Module
public class UploadActivityModule {
@ActivityScope
@Provides
UploadPresenter uploadPresenter() {
return new UploadPresenter();
}
}
如果是使用1.3.2方式提供实例,则在类上方添加作用域注解
@ActivityScope
public class UploadPresenter{
@Inject
public UploadPresenter() {
}
}
经过@Scope处理后,UploadActivity中的UploadPresenter实例将保持唯一。
全局单例,相信大家就很熟悉了,就是平时用饿汉懒汉等等写的那种单例模式。
前面已经说过@Scope作用是使同一个Component中供应的实例保持唯一。
也就是说,如果我在另一个Activity中再创建了一个新的Component,那么它所提供的UploadPresenter实例也将是新的。这和我们理解的全局单例并不一样。
所以,要想实现全局单例,那就要确保获取实例的Component一直都是同一个。
如何实现呢?答案是创建一个Component用于提供全局单例的实例,然后在Application中对该Component进行初始化,以后要获取单例时,都统一通过它来获取。
全局性的实例供应端
@Module
public class AppModule {
private Application mApplication;
public AppModule (Application application){
mApplication = application;
}
@Singleton
@Provides
Application application(){
return mApplication;
}
@Singleton
@Provides
ActivityStackManager activityStackManager() {
return new ActivityStackManager();
}
}
全局性的桥梁
@Singleton
@Component(modules = AppModule.class)
public interface AppComponent {
ActivityStackManager activityStackManager();
Application application();
}
Application中初始化
public class MyApplication extends Application {
public static AppComponent mAppComponent;
@Override
public void onCreate() {
super.onCreate();
mAppComponent= DaggerAppComponent.builder()
.appModule(new AppModule(this))
.build();
}
}
每次都通过Application中的AppComponent获取某实例,即可保证全局单例
public class UploadActivity extends AppCompatActivity{
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
MyApplication.mAppComponent.activityStackManager().pushOneActivity(this);
}
@Override
protected void onDestroy() {
super.onDestroy();
MyApplication.mAppComponent.activityStackManager().popOneActivity(this);
}
}
假如你并不希望在调用inject()方法后,就对@Inject标注的实例进行初始化注入,而是希望在用到该实例的时候再进行初始化,那么我们就可以使用Lazy和Provider来实现。
举例说明(省略实例供应端和桥梁的代码)
public class UploadActivity extends AppCompatActivity{
@Inject
Lazy mPresenter1;
@Inject
Provider mPresenter2;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
//注入实例
DaggerUploadActivityComponent.builder()
.build()
.inject(this);
}
public void xxx(){
//调用Lazy的get()方法后才开始初始化Presenter并得到该实例
//并且后面每次调用get()方法得到的实例都将是同一个。
mPresenter1.get().xxx();
//调用Provider的get()方法后才开始初始化Presenter并得到该实例
//并且后面每次调用get()方法,都会重新调用供应端的供应方法来获取新实例。
mPresenter2.get().xxx();
}
}
注意: 如果使用了前面介绍的作用域注解@Scope控制了实例的唯一性,那么即使多次调用Provider的get()方法,得到的依然是同一个实例。
假设供应端某个实例的初始化过程,需要用到全局的Context(即Application),那么我们可以通过桥梁间的依赖或包含,从全局的AppComponent中获取。
@Module
public class TestActivityModule {
//需要使用到Application参数
@Provides
DbHelper dbHelper(Application application){
return new DbHelper(application);
}
}
通过dependencies = xxx.classs指定要依赖的Component :
“` java
@Component(modules = TestActivityModule.class, dependencies = AppComponent.class)
public interface TestActivityComponent {
void inject(TestActivity testActivity);
}
被依赖的Component中需定义相关实例的获取方法 :
``` java
@Singleton
@Component(modules = AppModule.class)
public interface AppComponent {
Application application();
...
}
"se-preview-section-delimiter">
初始化TestActivityComponent时需传入依赖的Component
public class TestActivity extends AppCompatActivity{
@Inject
DbHelper mDbHelper;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
DaggerTestActivityComponent.builder()
.appComponent(MyApplication.mAppComponent) //传入依赖的Component
.build()
.inject(this);
}
}
"se-preview-section-delimiter">
TestActivityComponent使用@Subcomponent注解而不是@Component
@Subcomponent(modules = TestActivityModule.class)
public interface TestActivityComponent {
void inject(TestActivity testActivity);
}
"se-preview-section-delimiter">
AppComponent中定义相关方法,用来包含和获取SubComponent
@Singleton
@Component(modules = AppModule.class)
public interface AppComponent {
TestActivityComponent addSub(TestActivityModule testActivityModule);
...
}
"se-preview-section-delimiter">
通过AppComponent来获取SubComponent,然后注入
public class TestActivity extends AppCompatActivity{
@Inject
DbHelper mDbHelper;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
TestActivityComponent testActivityComponent = MyApplication.mAppComponent.addSub(new TestActivityModule());
testActivityComponent .inject(this);
}
}
7.1
在1.3.1和1.3.2介绍了两种实例供应方式:使用Module(方式一)、使用@Inject标注构造函数(方式二)。
那什么时候应该使用哪种方式呢? 假设现在供应端需要提供A类的实例
- 当无法在A类的构造函数上加入@Inject时(比如一些第三方库里的类),则使用方式一提供A实例。
- 当你希望在A类实例初始化时,A类中被@Inject标注的变量也被自动注入,则使用方式二提供A实例。
7.2
Module类可以声明为abstract抽象,但相关的供应方法需声明为static静态方法。
7.3
如果module中的供应方法声明了@Scope,那么它所属的component必须也声明相同的@Scope。
但如果component声明了@Scope,它的module的供应方法并不一定全都要声明@Scope。
7.4
@inject标注的实例变量不能声明为private,也不能为static,否则会编译报错。