我正在尝试在继承ActivityInstrumentationTestCase2的Android功能测试中使用Dagger。
设置代码如下所示:
@Override
protected void setUp() {
// TODO Auto-generated method stub
try {
super.setUp();
} catch (Exception e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
ObjectGraph.create(new TestModule()).inject(this);
this.activity = super.getActivity();
}`
通过调用super.getActivity()
触发的OnCreate方法不使用TestModule提供的类。但是,如果我手动运行我的活动(在测试上下文之外),那么我的非测试模块将提供/注入所有适当的类。
答案 0 :(得分:3)
我找到了一种方法,通过懒惰地创建对象图,将Dagger与ActivityInstrumentationTestCase2
一起使用。我所做的是等到创建对象图,直到第一次想要注入一个类,所以只要你在调用getActivity()
之前添加你的模块(它启动被测活动的活动生命周期)和在测试模块中使用overrides = true
,这将有效。这是相关的类和片段:
GraphHolder
,顾名思义,为我们保留ObjectGraph
对象。我们将对此课程进行所有调用,而不是直接调用ObjectGraph
。
public class GraphHolder {
private static GraphHolder sInstance;
private Object[] mModules;
private ObjectGraph mGraph;
private GraphHolder() {
}
public static GraphHolder getInstance() {
if (sInstance == null) {
sInstance = new GraphHolder();
}
return sInstance;
}
public void inject(Object object) {
if (mGraph == null) {
create();
}
mGraph.inject(object);
}
public <T> T get(Class<T> type) {
if (mGraph == null) {
create();
}
return mGraph.get(type);
}
public void addModules(Object... modules) {
if (mGraph != null) {
mGraph.plus(modules);
} else {
if (mModules == null) {
mModules = modules;
} else {
mModules = concatenate(mModules, modules);
}
}
}
private void create() {
mGraph = ObjectGraph.create(mModules);
mModules = null;
}
private Object[] concatenate(Object[] a, Object[] b) {
int aLength = a.length;
int bLength = b.length;
Object[] c = new Object[aLength + bLength];
System.arraycopy(a, 0, c, 0, aLength);
System.arraycopy(b, 0, c, aLength, bLength);
return c;
}
}
我们将在Application
类中添加我们的模块:
public class MyApplication extends Application {
@Override
public void onCreate() {
super.onCreate();
GraphHolder.getInstance().addModules(getModules());
}
Object[] getModules() {
return new Object[]{
// your modules here
};
}
}
在我们想要注入的类中,我们只需调用GraphHolder.getInstance().inject(this)
而不是ObjectGraph.inject(this)
在我们的测试模块中,我们将提供我们要覆盖的对象以进行测试,并将overrides = true
添加到@Module
注释中。如果发生冲突,这会告诉对象图更喜欢这个模块的提供者而不是其他人。
然后,在我们的测试中:
@Inject Foo mFoo;
@Override
public void setUp() {
super.setUp();
GraphHolder.getInstance().addModules(new TestFooModule());
GraphHolder.getInstance().inject(this); // This is when the object graph will be created
}
答案 1 :(得分:1)
ObjectGraph.create(new TestModule()).inject(this);
此代码尝试将TestModule创建的依赖项注入TestCase而不是测试的Activity。你在这里要做的是
ObjectGraph.create(new TestModule()).inject(this.activity);