当我尝试在测试中创建带有模拟Context
的AppCompatImageView
时,我收到了NullPointerException
。对正常ImageView
做同样的事情是有效的。
此测试通过:
import android.content.Context;
import android.widget.ImageView;
import org.junit.Before;
import org.junit.Test;
import org.junit.runner.RunWith;
import org.mockito.Mock;
import org.mockito.MockitoAnnotations;
import org.mockito.junit.MockitoJUnitRunner;
import static junit.framework.Assert.assertNotNull;
@RunWith(MockitoJUnitRunner.class)
public class ParallaxViewTest {
@Mock
Context mContext;
@Before
public void setup() {
MockitoAnnotations.initMocks(this);
}
@Test
public void initWithContext() throws Exception {
assertNotNull(mContext);
ImageView imageView = new ImageView(mContext);
// AppCompatImageView imageView = new AppCompatImageView(mContext);
}
}
此测试未通过:
import android.content.Context;
import android.support.v7.widget.AppCompatImageView;
import org.junit.Before;
import org.junit.Test;
import org.junit.runner.RunWith;
import org.mockito.Mock;
import org.mockito.MockitoAnnotations;
import org.mockito.junit.MockitoJUnitRunner;
import static junit.framework.Assert.assertNotNull;
@RunWith(MockitoJUnitRunner.class)
public class ParallaxViewTest {
@Mock
Context mContext;
@Before
public void setup() {
MockitoAnnotations.initMocks(this);
}
@Test
public void initWithContext() throws Exception {
assertNotNull(mContext);
// ImageView imageView = new ImageView(mContext);
AppCompatImageView imageView = new AppCompatImageView(mContext);
}
}
这是崩溃报告:
java.lang.NullPointerException
at android.support.v7.widget.ResourcesWrapper.<init>(ResourcesWrapper.java:46)
at android.support.v7.widget.TintResources.<init>(TintResources.java:34)
at android.support.v7.widget.TintContextWrapper.<init>(TintContextWrapper.java:100)
at android.support.v7.widget.TintContextWrapper.wrap(TintContextWrapper.java:68)
at android.support.v7.widget.AppCompatImageView.<init>(AppCompatImageView.java:60)
at android.support.v7.widget.AppCompatImageView.<init>(AppCompatImageView.java:56)
at android.support.v7.widget.AppCompatImageView.<init>(AppCompatImageView.java:52)
at example.views.ParallaxViewTest.initWithContext(ParallaxViewTest.java:30)
at sun.reflect.NativeMethodAccessorImpl.invoke0(Native Method)
at sun.reflect.NativeMethodAccessorImpl.invoke(NativeMethodAccessorImpl.java:62)
at sun.reflect.DelegatingMethodAccessorImpl.invoke(DelegatingMethodAccessorImpl.java:43)
at java.lang.reflect.Method.invoke(Method.java:497)
at org.junit.runners.model.FrameworkMethod$1.runReflectiveCall(FrameworkMethod.java:50)
at org.junit.internal.runners.model.ReflectiveCallable.run(ReflectiveCallable.java:12)
at org.junit.runners.model.FrameworkMethod.invokeExplosively(FrameworkMethod.java:47)
at org.junit.internal.runners.statements.InvokeMethod.evaluate(InvokeMethod.java:17)
at org.junit.internal.runners.statements.RunBefores.evaluate(RunBefores.java:26)
at org.junit.runners.ParentRunner.runLeaf(ParentRunner.java:325)
at org.junit.runners.BlockJUnit4ClassRunner.runChild(BlockJUnit4ClassRunner.java:78)
at org.junit.runners.BlockJUnit4ClassRunner.runChild(BlockJUnit4ClassRunner.java:57)
at org.junit.runners.ParentRunner$3.run(ParentRunner.java:290)
at org.junit.runners.ParentRunner$1.schedule(ParentRunner.java:71)
at org.junit.runners.ParentRunner.runChildren(ParentRunner.java:288)
at org.junit.runners.ParentRunner.access$000(ParentRunner.java:58)
at org.junit.runners.ParentRunner$2.evaluate(ParentRunner.java:268)
at org.junit.runners.ParentRunner.run(ParentRunner.java:363)
at org.mockito.internal.runners.DefaultInternalRunner$1.run(DefaultInternalRunner.java:68)
at org.mockito.internal.runners.DefaultInternalRunner.run(DefaultInternalRunner.java:74)
at org.mockito.internal.runners.StrictRunner.run(StrictRunner.java:39)
at org.mockito.junit.MockitoJUnitRunner.run(MockitoJUnitRunner.java:161)
at org.junit.runner.JUnitCore.run(JUnitCore.java:137)
at com.intellij.junit4.JUnit4IdeaTestRunner.startRunnerWithArgs(JUnit4IdeaTestRunner.java:117)
at com.intellij.junit4.JUnit4IdeaTestRunner.startRunnerWithArgs(JUnit4IdeaTestRunner.java:42)
at com.intellij.rt.execution.junit.JUnitStarter.prepareStreamsAndStart(JUnitStarter.java:262)
at com.intellij.rt.execution.junit.JUnitStarter.main(JUnitStarter.java:84)
at sun.reflect.NativeMethodAccessorImpl.invoke0(Native Method)
at sun.reflect.NativeMethodAccessorImpl.invoke(NativeMethodAccessorImpl.java:62)
at sun.reflect.DelegatingMethodAccessorImpl.invoke(DelegatingMethodAccessorImpl.java:43)
at java.lang.reflect.Method.invoke(Method.java:497)
at com.intellij.rt.execution.application.AppMain.main(AppMain.java:147)
这里是库:
testCompile 'junit:junit:4.12'
testCompile 'org.mockito:mockito-core:2.7.18'
我该如何解决这个问题?
编辑
如何获得带有资源的模拟Context
?
此测试未通过:
@Test
public void initWithContext() throws Exception {
assertNotNull(mContext); // PASS
assertNotNull(mContext.getResources()); // DO NOT PASS
// ImageView imageView = new ImageView(mContext);
// AppCompatImageView imageView = new AppCompatImageView(mContext);
}
当您转到堆栈跟踪 (ResourcesWrapper) 中列出的类的源代码时,您会发现:
public ResourcesWrapper(Resources resources) {
super(resources.getAssets(), resources.getDisplayMetrics(),
第 46 行是带有 super() 的行。
进一步查看堆栈跟踪中的类,您可能会遇到:
private TintContextWrapper(@NonNull final Context base) {
super(base);
...
mResources = new VectorEnabledTintResources(this, base.getResources());
所以,长话短说,是的,你提供了一个非空模拟对象来new AppCompatImageView()
你的代码。但是,您正在调用的代码正在该模拟对象上调用方法。当然,这就是您首先创建模拟的原因。但你猜怎么着;默认情况下,模拟框架将为任何方法调用返回NULL。
换句话说:你必须了解哪些调用将发生在该模拟上;这样你就可以准备模拟来返回一些非空的东西!
准确地说:我并不是说TintContextWrapper()中的那行导致了这个NPE;我主要是说:当你把一个模拟的对象交给其他代码时,你必须准备这个模拟,以在那些将要发生的方法调用上返回合理的结果。这很可能意味着您必须创建更多模拟;这样,类似于mockedContext.getResources()
的东西确实返回非空结果。
换句话说:你必须
- 识别在该模拟对象上发生的调用
- 然后,您必须确保这些调用将返回非 null(例如,通过再次返回模拟对象)。
除此之外:更有可能的是,真正的答案是使用Android特定的模拟框架。准备你的模拟让他们"做正确的事"很容易变成很多工作。
也许简单的答案是使用来自 mockito 的深存根,只需写
@Mock (answer = Answers.RETURNS_DEEP_STUBS)
但是你需要阅读/尝试;我自己没有用过。
并考虑到您的最新:您需要配置您的模拟,例如
when(context.getResources()).thenReturn(someOtherMock);
例如!这就是模拟的重点:您可以控制调用方法时会发生什么!