使用any时Mockito NullPointerException

时间:2016-03-28 23:12:11

标签: mockito junit4

我试图像这样使用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)

正如你所看到的,它是在给定的方法中(这个方法只有这段代码),所以我可以看到调试时,它到达那里并崩溃。

3 个答案:

答案 0 :(得分:6)

正如奥利弗在评论中提到的那样,你不能将when应用于所有对象。 Mockito通过子类化工作,因此您必须使用mockspy@Mock@Spy注释创建模拟实例;定制行为;然后安装模拟using dependency injection or other similar tricks

至于为什么会发生这种情况,any()的返回值实际上是null;像any这样的匹配器只能用作whenverify的参数,而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)

对我来说,真正的问题是我试图模拟一个间谍。模拟间谍时,您必须使用 doReturndoAnswer 方法。否则,间谍的方法将在尝试模拟它时被实际调用,并且可能会发生意外行为。例如。当使用 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"));
    }

}