使用任何时 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 通过子类化工作,因此您必须使用 mockspy@Mock@Spy 注释创建模拟实例;自定义行为;然后安装模拟 using dependency injection or other similar tricks.


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

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

}