我试图像这样使用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)
我这样做的原因是因为我到达代码的这一点时,有大量的对象被创建。嘲笑每一步都需要时间。
当代码到达此mockito语句时,我获得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$1.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$1.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$3.run(ParentRunner.java:238)
at org.junit.runners.ParentRunner$1.schedule(ParentRunner.java:63)
at org.junit.runners.ParentRunner.runChildren(ParentRunner.java:236)
at org.junit.runners.ParentRunner.access$000(ParentRunner.java:53)
at org.junit.runners.ParentRunner$2.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)
正如你所看到的,它是在给定的方法中(这个方法只有这段代码),所以我可以看到调试时,它到达那里并崩溃。
答案 0 :(得分:6)
正如奥利弗在评论中提到的那样,你不能将when
应用于所有对象。 Mockito通过子类化工作,因此您必须使用mock
,spy
或@Mock
或@Spy
注释创建模拟实例;定制行为;然后安装模拟using dependency injection or other similar tricks。
至于为什么会发生这种情况,any()
的返回值实际上是null
;像any
这样的匹配器只能用作when
和verify
的参数,而Mockito不能生成代表&#34的Class
的专用实例;任何Class&#34;,因此Mockito返回一个虚拟值(null)并将数据存储在一个专门的参数匹配器堆栈中。虽然Mockito有更好的错误消息提醒您这种情况,但您在Mockito之前的代码NPE可以通过使用示例为您提供正确的例外。
有关匹配器返回值和堆栈的更多信息,请参阅"How do Mockito matchers work?"上的其他SO答案。
答案 1 :(得分:0)
要扩展@ jeff-bowman在回答中所说的内容,any()返回null,因此,如果您需要不返回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>()
答案 2 :(得分:0)
对我来说,真正的问题是我试图模拟一个间谍。模拟间谍时,您必须使用 doReturn
或 doAnswer
方法。否则,间谍的方法将在尝试模拟它时被实际调用,并且可能会发生意外行为。例如。当使用 when
调用 any()
时,any()
只会返回 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"));
}
}