NullPointerException 使用模拟上下文创建 AppCompatImageView
NullPointerException creating an AppCompatImageView with mock Context
当我尝试在测试中使用模拟 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.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.run(ParentRunner.java:290)
at org.junit.runners.ParentRunner.schedule(ParentRunner.java:71)
at org.junit.runners.ParentRunner.runChildren(ParentRunner.java:288)
at org.junit.runners.ParentRunner.access[=12=]0(ParentRunner.java:58)
at org.junit.runners.ParentRunner.evaluate(ParentRunner.java:268)
at org.junit.runners.ParentRunner.run(ParentRunner.java:363)
at org.mockito.internal.runners.DefaultInternalRunner.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) 中列出的 class 的源代码时,您会发现:
public ResourcesWrapper(Resources resources) {
super(resources.getAssets(), resources.getDisplayMetrics(),
第 46 行是带有 super() 的那一行。
进一步查看堆栈跟踪中的 classes,您可能会发现:
private TintContextWrapper(@NonNull final Context base) {
super(base);
...
mResources = new VectorEnabledTintResources(this, base.getResources());
所以,长话短说,是的,您在代码中向 new AppCompatImageView()
提供了一个 非空 模拟对象。但是你调用的代码 正在调用 模拟的 对象上的 方法。当然,这就是您首先创建模拟的原因。但猜猜怎么了;默认情况下,模拟框架将为任何方法调用 return null。
换句话说:您必须了解哪些 调用将在该模拟上发生;这样你就可以准备模拟 return 东西 非空 了!
准确地说:我并不是说 TintContextWrapper() 中的那一行恰好导致了这个 NPE;我主要是说:当您将模拟对象提供给其他代码时,您必须 准备 模拟以 return 那些将要发生的方法调用的合理结果。这很可能意味着您必须创建更多模拟;所以像 mockedContext.getResources()
这样的东西会 return 一个 non-null 结果。
换句话说:你必须
- 识别在该模拟对象上发生的那些调用
- 然后你必须确保这些调用将 return 非空(例如通过再次 returning 模拟对象)。
除此之外:更有可能的是,真正的答案是使用 Android 特定的 模拟框架。准备模拟以获取它们 "do the right thing" 很容易变成大量工作。
也许 简单的答案是使用 mockito 中的 deep stubbing,只需编写
@Mock (answer = Answers.RETURNS_DEEP_STUBS)
但是你需要read/try那个;我自己没用过。
并给出你的最新消息:你需要配置你的模拟,比如
when(context.getResources()).thenReturn(someOtherMock);
例如!这就是 mock 的 全部 点:您可以控制调用方法时发生的事情!
当我尝试在测试中使用模拟 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.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.run(ParentRunner.java:290)
at org.junit.runners.ParentRunner.schedule(ParentRunner.java:71)
at org.junit.runners.ParentRunner.runChildren(ParentRunner.java:288)
at org.junit.runners.ParentRunner.access[=12=]0(ParentRunner.java:58)
at org.junit.runners.ParentRunner.evaluate(ParentRunner.java:268)
at org.junit.runners.ParentRunner.run(ParentRunner.java:363)
at org.mockito.internal.runners.DefaultInternalRunner.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) 中列出的 class 的源代码时,您会发现:
public ResourcesWrapper(Resources resources) {
super(resources.getAssets(), resources.getDisplayMetrics(),
第 46 行是带有 super() 的那一行。
进一步查看堆栈跟踪中的 classes,您可能会发现:
private TintContextWrapper(@NonNull final Context base) {
super(base);
...
mResources = new VectorEnabledTintResources(this, base.getResources());
所以,长话短说,是的,您在代码中向 new AppCompatImageView()
提供了一个 非空 模拟对象。但是你调用的代码 正在调用 模拟的 对象上的 方法。当然,这就是您首先创建模拟的原因。但猜猜怎么了;默认情况下,模拟框架将为任何方法调用 return null。
换句话说:您必须了解哪些 调用将在该模拟上发生;这样你就可以准备模拟 return 东西 非空 了!
准确地说:我并不是说 TintContextWrapper() 中的那一行恰好导致了这个 NPE;我主要是说:当您将模拟对象提供给其他代码时,您必须 准备 模拟以 return 那些将要发生的方法调用的合理结果。这很可能意味着您必须创建更多模拟;所以像 mockedContext.getResources()
这样的东西会 return 一个 non-null 结果。
换句话说:你必须
- 识别在该模拟对象上发生的那些调用
- 然后你必须确保这些调用将 return 非空(例如通过再次 returning 模拟对象)。
除此之外:更有可能的是,真正的答案是使用 Android 特定的 模拟框架。准备模拟以获取它们 "do the right thing" 很容易变成大量工作。
也许 简单的答案是使用 mockito 中的 deep stubbing,只需编写
@Mock (answer = Answers.RETURNS_DEEP_STUBS)
但是你需要read/try那个;我自己没用过。
并给出你的最新消息:你需要配置你的模拟,比如
when(context.getResources()).thenReturn(someOtherMock);
例如!这就是 mock 的 全部 点:您可以控制调用方法时发生的事情!