使用任何时 Mockito NullPointerException
Mockito NullPointerException while using any
我正在尝试像这样使用 Mockito :
Mockito.when(Mockito.any(ObjectMapper.class).readValue(Mockito.any(BufferedReader.class),Mockito.any(Class.class))).thenReturn(new Person("1","abc"));
这是杰克逊图书馆的。
public <T> T readValue(Reader src, Class<T> valueType)
我这样做的原因是当我到达代码的这一点时,已经创建了大量对象。嘲笑每一步都需要时间。
当代码到达这个模拟语句时我得到 NPE 的任何原因?
堆栈跟踪:
java.lang.NullPointerException
at com.prashant.flax.ShellTest.givenDirectoryHasFiles(ShellTest.java:139)
at com.prashant.flax.ShellTest.testExecute(ShellTest.java:61)
at sun.reflect.NativeMethodAccessorImpl.invoke0(Native Method)
at sun.reflect.NativeMethodAccessorImpl.invoke(NativeMethodAccessorImpl.java:62)
at sun.reflect.DelegatingMethodAccessorImpl.invoke(DelegatingMethodAccessorImpl.java:43)
at org.junit.runners.model.FrameworkMethod.runReflectiveCall(FrameworkMethod.java:47)
at org.junit.internal.runners.model.ReflectiveCallable.run(ReflectiveCallable.java:12)
at org.junit.runners.model.FrameworkMethod.invokeExplosively(FrameworkMethod.java:44)
at org.junit.internal.runners.statements.InvokeMethod.evaluate(InvokeMethod.java:17)
at org.junit.rules.ExternalResource.evaluate(ExternalResource.java:48)
at org.junit.rules.RunRules.evaluate(RunRules.java:20)
at org.junit.runners.ParentRunner.runLeaf(ParentRunner.java:271)
at org.junit.runners.BlockJUnit4ClassRunner.runChild(BlockJUnit4ClassRunner.java:70)
at org.junit.runners.BlockJUnit4ClassRunner.runChild(BlockJUnit4ClassRunner.java:50)
at org.junit.runners.ParentRunner.run(ParentRunner.java:238)
at org.junit.runners.ParentRunner.schedule(ParentRunner.java:63)
at org.junit.runners.ParentRunner.runChildren(ParentRunner.java:236)
at org.junit.runners.ParentRunner.access[=12=]0(ParentRunner.java:53)
at org.junit.runners.ParentRunner.evaluate(ParentRunner.java:229)
at org.junit.runners.ParentRunner.run(ParentRunner.java:309)
at org.junit.runner.JUnitCore.run(JUnitCore.java:160)
at com.intellij.junit4.JUnit4IdeaTestRunner.startRunnerWithArgs(JUnit4IdeaTestRunner.java:78)
at com.intellij.rt.execution.junit.JUnitStarter.prepareStreamsAndStart(JUnitStarter.java:212)
at com.intellij.rt.execution.junit.JUnitStarter.main(JUnitStarter.java:68)
at sun.reflect.NativeMethodAccessorImpl.invoke0(Native Method)
at sun.reflect.NativeMethodAccessorImpl.invoke(NativeMethodAccessorImpl.java:62)
at com.intellij.rt.execution.application.AppMain.main(AppMain.java:140)
如您所见,它在给定的方法中(此方法只有这段代码),所以我在调试时可以看到,它到达那里并崩溃了。
正如 Oliver 在评论中提到的,您不能将 when
应用于所有对象。 Mockito 通过子类化工作,因此您必须使用 mock
、spy
或 @Mock
或 @Spy
注释创建模拟实例;自定义行为;然后安装模拟 using dependency injection or other similar tricks.
至于为什么会发生这种情况,any()
的return值实际上是null
;像 any
这样的匹配器只能用作 when
和 verify
的参数,Mockito 不能生成代表 "any Class" 的 Class
的专门实例,因此 Mockito return 是一个虚拟值(null)并将数据存储在一个专门的参数匹配器堆栈中。尽管 Mockito 有更好的错误消息来提醒您注意这种情况,但 Mockito 之前的代码 NPE 可以通过使用示例为您提供适当的异常。
有关匹配器 return 值和堆栈的更多信息,请参阅我在 "How do Mockito matchers work?" 上的其他 SO 回答。
为了扩展@jeff-bowman 在他的回答中所说的,any() returns null,所以如果你需要一些不 return null 的东西,你可以试试这个:
/**
* never returns null
*/
private inline fun <reified T> any(type: Class<T>): T = Mockito.any(type)
/**
* returns null
*/
private inline fun <reified T> any(): T = Mockito.any<T>()
对我来说,真正的问题是我试图嘲笑间谍。模拟间谍时,您必须使用 doReturn
或 doAnswer
方法。否则间谍的方法将在尝试模拟它时实际被调用,并且可能发生意外行为。例如。当用 any()
调用 when
时,any()
将只是 return null
。所以你很可能会得到一个 NullPointerException
.
这里有一个完整的例子来演示行为:
package com.company;
import org.junit.Test;
import static org.junit.Assert.assertEquals;
import static org.mockito.ArgumentMatchers.any;
import static org.mockito.ArgumentMatchers.anyString;
import static org.mockito.Mockito.*;
public class SpyTest {
public class Controller {
public String method(String arg) {
return arg.substring(0, 1);
}
}
@Test
public void withMockTest() {
Controller controllerMocked = mock(Controller.class);
when(controllerMocked.method(anyString())).thenReturn("42");
assertEquals("42", controllerMocked.method("FOO"));
}
@Test(expected = NullPointerException.class)
public void withSpyWhenThenReturnBreakingBecauseMethodToBeMockedIsActuallyBeingCalled() {
Controller controllerSpied = spy(new Controller());
when(controllerSpied.method(any())).thenReturn("42");
}
@Test
public void withSpyDoReturnWhen() {
Controller controllerSpied = spy(new Controller());
doReturn("42").when(controllerSpied).method(any());
assertEquals("42", controllerSpied.method("FOO"));
}
}
我正在尝试像这样使用 Mockito :
Mockito.when(Mockito.any(ObjectMapper.class).readValue(Mockito.any(BufferedReader.class),Mockito.any(Class.class))).thenReturn(new Person("1","abc"));
这是杰克逊图书馆的。
public <T> T readValue(Reader src, Class<T> valueType)
我这样做的原因是当我到达代码的这一点时,已经创建了大量对象。嘲笑每一步都需要时间。
当代码到达这个模拟语句时我得到 NPE 的任何原因?
堆栈跟踪:
java.lang.NullPointerException
at com.prashant.flax.ShellTest.givenDirectoryHasFiles(ShellTest.java:139)
at com.prashant.flax.ShellTest.testExecute(ShellTest.java:61)
at sun.reflect.NativeMethodAccessorImpl.invoke0(Native Method)
at sun.reflect.NativeMethodAccessorImpl.invoke(NativeMethodAccessorImpl.java:62)
at sun.reflect.DelegatingMethodAccessorImpl.invoke(DelegatingMethodAccessorImpl.java:43)
at org.junit.runners.model.FrameworkMethod.runReflectiveCall(FrameworkMethod.java:47)
at org.junit.internal.runners.model.ReflectiveCallable.run(ReflectiveCallable.java:12)
at org.junit.runners.model.FrameworkMethod.invokeExplosively(FrameworkMethod.java:44)
at org.junit.internal.runners.statements.InvokeMethod.evaluate(InvokeMethod.java:17)
at org.junit.rules.ExternalResource.evaluate(ExternalResource.java:48)
at org.junit.rules.RunRules.evaluate(RunRules.java:20)
at org.junit.runners.ParentRunner.runLeaf(ParentRunner.java:271)
at org.junit.runners.BlockJUnit4ClassRunner.runChild(BlockJUnit4ClassRunner.java:70)
at org.junit.runners.BlockJUnit4ClassRunner.runChild(BlockJUnit4ClassRunner.java:50)
at org.junit.runners.ParentRunner.run(ParentRunner.java:238)
at org.junit.runners.ParentRunner.schedule(ParentRunner.java:63)
at org.junit.runners.ParentRunner.runChildren(ParentRunner.java:236)
at org.junit.runners.ParentRunner.access[=12=]0(ParentRunner.java:53)
at org.junit.runners.ParentRunner.evaluate(ParentRunner.java:229)
at org.junit.runners.ParentRunner.run(ParentRunner.java:309)
at org.junit.runner.JUnitCore.run(JUnitCore.java:160)
at com.intellij.junit4.JUnit4IdeaTestRunner.startRunnerWithArgs(JUnit4IdeaTestRunner.java:78)
at com.intellij.rt.execution.junit.JUnitStarter.prepareStreamsAndStart(JUnitStarter.java:212)
at com.intellij.rt.execution.junit.JUnitStarter.main(JUnitStarter.java:68)
at sun.reflect.NativeMethodAccessorImpl.invoke0(Native Method)
at sun.reflect.NativeMethodAccessorImpl.invoke(NativeMethodAccessorImpl.java:62)
at com.intellij.rt.execution.application.AppMain.main(AppMain.java:140)
如您所见,它在给定的方法中(此方法只有这段代码),所以我在调试时可以看到,它到达那里并崩溃了。
正如 Oliver 在评论中提到的,您不能将 when
应用于所有对象。 Mockito 通过子类化工作,因此您必须使用 mock
、spy
或 @Mock
或 @Spy
注释创建模拟实例;自定义行为;然后安装模拟 using dependency injection or other similar tricks.
至于为什么会发生这种情况,any()
的return值实际上是null
;像 any
这样的匹配器只能用作 when
和 verify
的参数,Mockito 不能生成代表 "any Class" 的 Class
的专门实例,因此 Mockito return 是一个虚拟值(null)并将数据存储在一个专门的参数匹配器堆栈中。尽管 Mockito 有更好的错误消息来提醒您注意这种情况,但 Mockito 之前的代码 NPE 可以通过使用示例为您提供适当的异常。
有关匹配器 return 值和堆栈的更多信息,请参阅我在 "How do Mockito matchers work?" 上的其他 SO 回答。
为了扩展@jeff-bowman 在他的回答中所说的,any() returns null,所以如果你需要一些不 return null 的东西,你可以试试这个:
/**
* never returns null
*/
private inline fun <reified T> any(type: Class<T>): T = Mockito.any(type)
/**
* returns null
*/
private inline fun <reified T> any(): T = Mockito.any<T>()
对我来说,真正的问题是我试图嘲笑间谍。模拟间谍时,您必须使用 doReturn
或 doAnswer
方法。否则间谍的方法将在尝试模拟它时实际被调用,并且可能发生意外行为。例如。当用 any()
调用 when
时,any()
将只是 return null
。所以你很可能会得到一个 NullPointerException
.
这里有一个完整的例子来演示行为:
package com.company;
import org.junit.Test;
import static org.junit.Assert.assertEquals;
import static org.mockito.ArgumentMatchers.any;
import static org.mockito.ArgumentMatchers.anyString;
import static org.mockito.Mockito.*;
public class SpyTest {
public class Controller {
public String method(String arg) {
return arg.substring(0, 1);
}
}
@Test
public void withMockTest() {
Controller controllerMocked = mock(Controller.class);
when(controllerMocked.method(anyString())).thenReturn("42");
assertEquals("42", controllerMocked.method("FOO"));
}
@Test(expected = NullPointerException.class)
public void withSpyWhenThenReturnBreakingBecauseMethodToBeMockedIsActuallyBeingCalled() {
Controller controllerSpied = spy(new Controller());
when(controllerSpied.method(any())).thenReturn("42");
}
@Test
public void withSpyDoReturnWhen() {
Controller controllerSpied = spy(new Controller());
doReturn("42").when(controllerSpied).method(any());
assertEquals("42", controllerSpied.method("FOO"));
}
}