[Android]在Dagger 2中Activities和Subcomponents的多绑定(翻译)
以下内容为原创,欢迎转载,转载请注明
来自天天博客:http://www.cnblogs.com/tiantianbyconan/p/6266442.html
在Dagger 2中Activities和Subcomponents的多绑定
原文:http://frogermcs.github.io/activities-multibinding-in-dagger-2
几个月前,在MCE^3会议中,Gregory Kick在他的演讲中展示了一个提供Subcomponents(比如,为Activity)的新概念。新的方式给我们带来了一个创建不使用AppComponent对象引用(以前时Activities Subcomponents的工厂)的方式。为了让它成为现实,我们不得不等到了新的Dagger release版本:version 2.7。
问题
在Dagger 2.7之前,创建Subcomponent(比如,AppComponent
的subcomponent MainActivityComponent
)我们必须要在父Component中声明它的工厂:
@Singleton
@Component(
modules = {
AppModule.class
}
)
public interface AppComponent {
MainActivityComponent plus(MainActivityComponent.ModuleImpl module);
//...
}
多亏Dagger理解这个声明,MainActivityComponent
能够访问从AppComponent
的依赖。
有了这个,MainActivity
中的注入看起来如下:
@Override
protected ActivityComponent onCreateComponent() {
((MyApplication) getApplication()).getComponent().plus(new MainActivityComponent.ModuleImpl(this));
component.inject(this);
return component;
}
这个代码的问题在于:
Activity依赖于
AppComponent
(通过((MyApplication) getApplication()).getComponent())
返回 - 我们是否想要去创建Subcomponent,我们需要去访问父Component的对象)。AppComponent
必须要去声明所有Subcomponents(或者它们的builders),比如:MainActivityComponent plus(MainActivityComponent.ModuleImpl module);
。
Modules.subcomponents
从Dagger 2.7开始,我们有了一个新的方法来声明subcomponents的父级。@Module
注解有一个可选的subcomponents属性,它可以得到subcomponents类的列表,它们应该是安装此module组件的子component。
Example:
@Module(
subcomponents = {
MainActivityComponent.class,
SecondActivityComponent.class
})
public abstract class ActivityBindingModule {
//...
}
ActivityBindingModule
在AppComponent
中被安装。这表示MainActivityComponent
和SecondActivityComponent
两者都是AppComponent
的Subcomponents。
Subcomponents的声明在这种方法中不需要明确地在AppComponent
中进行声明(就像本章开头的代码)。
Activities的多绑定
让我们来看看我们怎么样使用Modules.subcomponents
来构建Activities多绑定并且摆脱AppComponent对象传入Activity(这在这个演讲中也解释到)。我将只浏览代码中最重要的部分。整个实现已在Github中可用:Dagger2Recipes-ActivitiesMultibinding。
我们的app包含两个屏幕:MainActivity
和SecondActivity
。我们想要去给它们两者提供Subcomponents且并不传入AppComponent
对象。
让我们从为所有Activity Components builders构建一个基本的接口来开始:
public interface ActivityComponentBuilder<M extends ActivityModule, C extends ActivityComponent> {
ActivityComponentBuilder<M, C> activityModule(M activityModule);
C build();
}
Subcomponents:MainActivityComponent
的例子看起来如下:
@ActivityScope
@Subcomponent(
modules = MainActivityComponent.MainActivityModule.class
)
public interface MainActivityComponent extends ActivityComponent<MainActivity> {
@Subcomponent.Builder
interface Builder extends ActivityComponentBuilder<MainActivityModule, MainActivityComponent> {
}
@Module
class MainActivityModule extends ActivityModule<MainActivity> {
MainActivityModule(MainActivity activity) {
super(activity);
}
}
}
现在我们可以使用Subcomponents builders的Map
来得到每一个Activity类的意图builder。让我们如下使用Multibinding
特性:
@Module(
subcomponents = {
MainActivityComponent.class,
SecondActivityComponent.class
})
public abstract class ActivityBindingModule {
@Binds
@IntoMap
@ActivityKey(MainActivity.class)
public abstract ActivityComponentBuilder mainActivityComponentBuilder(MainActivityComponent.Builder impl);
@Binds
@IntoMap
@ActivityKey(SecondActivity.class)
public abstract ActivityComponentBuilder secondActivityComponentBuilder(SecondActivityComponent.Builder impl);
}
ActivityBindingModule
在AppComponent
中被安装。就如它被解释的那样,多亏MainActivityComponent
和SecondActivityComponent
将会是AppComponent
的Subcomponent。
现在我们可以注入Subcomponents
builder的Map(比如,注入到MyApplication
class):
public class MyApplication extends Application implements HasActivitySubcomponentBuilders {
@Inject
Map<Class<? extends Activity>, ActivityComponentBuilder> activityComponentBuilders;
private AppComponent appComponent;
public static HasActivitySubcomponentBuilders get(Context context) {
return ((HasActivitySubcomponentBuilders) context.getApplicationContext());
}
@Override
public void onCreate() {
super.onCreate();
appComponent = DaggerAppComponent.create();
appComponent.inject(this);
}
@Override
public ActivityComponentBuilder getActivityComponentBuilder(Class<? extends Activity> activityClass) {
return activityComponentBuilders.get(activityClass);
}
}
我们创建了HashActivitySubcomponentBuilders
接口作为额外的抽象(因为builders的Map
不一定是注入到Appliction
类的):
public interface HasActivitySubcomponentBuilders {
ActivityComponentBuilder getActivityComponentBuilder(Class<? extends Activity> activityClass);
}
然后最后在Activity class中进行注入的实现:
public class MainActivity extends BaseActivity {
//...
@Override
protected void injectMembers(HasActivitySubcomponentBuilders hasActivitySubcomponentBuilders) {
((MainActivityComponent.Builder) hasActivitySubcomponentBuilders.getActivityComponentBuilder(MainActivity.class))
.activityModule(new MainActivityComponent.MainActivityModule(this))
.build().injectMembers(this);
}
}
它非常类似于我们的第一个实现,但如上,最重要的事是我们不再传入ActivityComponent
对象到我们的Activities中。
用例example —— instrumentation tests mocking
除了解耦和解决循环依赖(Activity <-> Application),这不是一个大的问题,尤其是在较小的项目/团队中,让我们思考一个这个实现有帮助的真实用例 —— 在instrumentation testing中的mocking依赖。
目前在Android Instrumentation测试中mocking依赖最著名的方式之一是使用DaggerMock(Github 项目地址)。虽然DaggerMock是一个强大的工具,但是非常难理解它面具之下是怎么工作的。其中有一些反射代码不容易被追踪。
在Activity中直接构建Subcomponent,而不需要访问AppComponent类给了我们一个方式来测试单独的Activity并从我们app的其它部分解耦。
听起来很酷,现在我们来看下代码。
在我们的instrumentation test中使用Applicaton类:
public class ApplicationMock extends MyApplication {
public void putActivityComponentBuilder(ActivityComponentBuilder builder, Class<? extends Activity> cls) {
Map<Class<? extends Activity>, ActivityComponentBuilder> activityComponentBuilders = new HashMap<>(this.activityComponentBuilders);
activityComponentBuilders.put(cls, builder);
this.activityComponentBuilders = activityComponentBuilders;
}
}
putActivityComponentBuilder()
方法给我们一个对给定Activity类替换ActivityComponentBuilder的实现的方法。
现在来看下我们Espresso Instrumentation Test例子:
@RunWith(AndroidJUnit4.class)
public class MainActivityUITest {
@Rule
public MockitoRule mockitoRule = MockitoJUnit.rule();
@Rule
public ActivityTestRule<MainActivity> activityRule = new ActivityTestRule<>(MainActivity.class, true, false);
@Mock
MainActivityComponent.Builder builder;
@Mock
Utils utilsMock;
private MainActivityComponent mainActivityComponent = new MainActivityComponent() {
@Override
public void injectMembers(MainActivity instance) {
instance.mainActivityPresenter = new MainActivityPresenter(instance, utilsMock);
}
};
@Before
public void setUp() {
when(builder.build()).thenReturn(mainActivityComponent);
when(builder.activityModule(any(MainActivityComponent.MainActivityModule.class))).thenReturn(builder);
ApplicationMock app = (ApplicationMock) InstrumentationRegistry.getTargetContext().getApplicationContext();
app.putActivityComponentBuilder(builder, MainActivity.class);
}
一步一步来:
我们提供了
MainActivityComponent.Builder
的Mock和所有我们必须要mock的依赖(在本例中只是Utils
)。我们mockedBuilder
返回一个MainActivityComponent
的一个自定义实现,它用于注入MainActivityPresenter
(其中使用了mockedUtils
)。然后我们的
MainActivityComponent.Builder
替换了在MyApplication
(28行)中被注入的原始Builder:app.putActivityComponentBuilder(builder, MainActivity.class);
最后测试 —— 我们mock
Util.getHardcodedText()
方法。注入过程发生在Activity被创建(36行):activityRule.launchActivity(new Intent());
接着在最后我们使用Espresso来检验结果。
以上就是全部。如你所见,几乎一切都发生在MainActivityUITest
类中,而且代码相当简单和可读。
源码
如果你想自己去测试这个实现,源码与工作例子展示怎么去创建Activities Multibinding和在Instrumentation Tests中mock依赖见Github:Dagger2Recipes-ActivitiesMultibinding
感谢阅读!
作者
Head of Mobile Development @ Azimo
> __[Android]使用Dagger 2依赖注入 - DI介绍(翻译):__
> __[Android]使用Dagger 2依赖注入 - API(翻译):__
> __[Android]使用Dagger 2依赖注入 - 自定义Scope(翻译):__
> __[Android]使用Dagger 2依赖注入 - 图表创建的性能(翻译):__
> __[Android]Dagger2Metrics - 测量DI图表初始化的性能(翻译):__
> __[Android]使用Dagger 2进行依赖注入 - Producers(翻译):__
> __[Android]在Dagger 2中使用RxJava来进行异步注入(翻译):__
> __[Android]使用Dagger 2来构建UserScope(翻译):__
> __[Android]在Dagger 2中Activities和Subcomponents的多绑定(翻译):__
[Android]在Dagger 2中Activities和Subcomponents的多绑定(翻译)的更多相关文章
- [Android]使用Dagger 2依赖注入 - 图表创建的性能(翻译)
以下内容为原创,欢迎转载,转载请注明 来自天天博客:http://www.cnblogs.com/tiantianbyconan/p/5098943.html 使用Dagger 2依赖注入 - 图表创 ...
- [Android]在Dagger 2中使用RxJava来进行异步注入(翻译)
以下内容为原创,欢迎转载,转载请注明 来自天天博客: # 在Dagger 2中使用RxJava来进行异步注入 > 原文: 几星期前我写了一篇关于在Dagger 2中使用*Producers*进行 ...
- Android 和 Dagger 2 中的依赖注入
原文:Dependency Injection in Android with Dagger 2 作者:Joe Howard 译者:kmyhy 在现代开发团队中到处充斥着"你一定要用依赖注入 ...
- [Android]使用Dagger 2来构建UserScope(翻译)
以下内容为原创,欢迎转载,转载请注明 来自天天博客:http://www.cnblogs.com/tiantianbyconan/p/6237731.html 使用Dagger 2来构建UserSco ...
- [Android]使用Dagger 2进行依赖注入 - Producers(翻译)
以下内容为原创,欢迎转载,转载请注明 来自天天博客:http://www.cnblogs.com/tiantianbyconan/p/6234811.html 使用Dagger 2进行依赖注入 - P ...
- [Android]使用Dagger 2依赖注入 - DI介绍(翻译)
以下内容为原创,欢迎转载,转载请注明 来自天天博客:http://www.cnblogs.com/tiantianbyconan/p/5092083.html 使用Dagger 2依赖注入 - DI介 ...
- [Android]使用Dagger 2依赖注入 - API(翻译)
以下内容为原创,欢迎转载,转载请注明 来自天天博客:http://www.cnblogs.com/tiantianbyconan/p/5092525.html 使用Dagger 2依赖注入 - API ...
- [Android]使用Dagger 2依赖注入 - 自定义Scope(翻译)
以下内容为原创,欢迎转载,转载请注明 来自天天博客:http://www.cnblogs.com/tiantianbyconan/p/5095426.html 使用Dagger 2依赖注入 - 自定义 ...
- Android 4.4(KitKat)中apk包的安装过程
原文地址:http://blog.csdn.net/jinzhuojun/article/details/25542011 事实上对于apk包的安装.4.4和之前版本号没大的区别. Android中a ...
随机推荐
- 远程连接Ucenter数据库
网站和Ucenter不是同一服务器的连接方法~我折腾了好几天,终于找到了这方法!各位连接不上的不妨试试~什么事只有试过才知道行不行! define('UC_CONNECT', 'mysql'); de ...
- iis7如何取消目录的可执行权限
我们需要把IIs中某一个目录的可执行权限去掉.这在IIs6中是非常方便的,可是因为iis7的机制小编也找了不少资料才找到. 第一步:先选择需要取消权限的目录,然后在右边可以看到 “处理程序映射” 双击 ...
- linux点滴:rsync
rsync(remote sync)是一款远程同步工具,可以实现全量备份.增量备份.本地备份.删除,核心功能是远程数据备份. 工作原理 rsync核心算法 1.分块checksum算法 首先,把文件平 ...
- EasyUI portal自定义小图标,不是用js方式加载
<script src="~/Scripts/jquery.portal.js"></script> <script> $(function ( ...
- ExtJS4加载FormPanel数据的几种方式
我们做web应用最多的就是处理表单,extjs为我们提供了很多处理表单的功能,很多初学者疑惑怎么加载表单数据,到底能用什么方式加载?本文中,我将我自己实验过的进行一下总结,自己备忘,也希望能帮助到其他 ...
- Python之创建单元素tuple
tuple和list一样,可以包含 0 个.1个和任意多个元素. 包含多个元素的 tuple,前面我们已经创建过了. 包含 0 个元素的 tuple,也就是空tuple,直接用 ()表示: >& ...
- 我的PHP之旅--SQL语句
SQL语句 结构化查询语言(Structured Query Language)简称SQL,是一种操作数据的语言. 增加记录 INSERT INTO table_name(字段1, 字段2, 字段3) ...
- Automotive Security的一些资料和心得(4):Automotive Safeguards
通常一辆汽车会包括超过80个ECUs.所有软件代码大小正在快速增加,将会超过1GB.软件protection是必不可少的. 1. 软件保护 1.1. 安全boot Software violating ...
- codeforces 388C Fox and Card Game
刚刚看到这个题感觉是博弈题: 不过有感觉不像,应该是个贪心: 于是就想贪心策略: 举了一个例子: 3 3 1 2 3 4 3 4 1 2 5 4 1 2 5 8 如果他们两个每次都拿对自己最有利的那个 ...
- JAVA客户端API调用memcached两种方式
1. memcached client for java客户端API:memcached client for java 引入jar包:java-memcached-2.6.2.jar package ...