如何模拟方法与无效返回类型?

我实现了一个观察者模式,但我不能用Mockito模拟它,因为我不知道怎么做。

我试图在网上找到一个例子,但没有成功。

我的类是这样的:

public class World {

    List<Listener> listeners;

    void addListener(Listener item) {
        listeners.add(item);
    }

    void doAction(Action goal,Object obj) {
        setState("i received");
        goal.doAction(obj);
        setState("i finished");
    }

    private string state;
    //setter getter state
} 

public class WorldTest implements Listener {

    @Test public void word{
    World  w= mock(World.class);
    w.addListener(this);
    ...
    ...

    }
}

interface Listener {
    void doAction();
}

系统不会被mock触发。

我想展示上述系统的状态。并据此断言。


当前回答

在Java 8中,假设你有一个org.mockito.Mockito.doAnswer的静态导入,这可以做得更清楚一些:

doAnswer(i -> {
  // Do stuff with i.getArguments() here
  return null;
}).when(*mock*).*method*(*methodArguments*);

返回null;是重要的,如果没有它,编译将失败,并出现一些相当模糊的错误,因为它将无法为doAnswer找到合适的覆盖。

例如,ExecutorService可以立即执行传递给execute()的任何可运行对象,可以使用以下方法实现:

doAnswer(i -> {
  ((Runnable) i.getArguments()[0]).run();
  return null;
}).when(executor).execute(any());

其他回答

如何用mockito模拟void方法-有两个选项:

doAnswer -如果我们想让我们的mock void方法做一些事情(尽管它是void,但仍然模拟该行为)。 doThrow——如果你想从mock void方法抛出一个异常,那么还有Mockito.doThrow()。

下面是如何使用它的示例(不是理想的用例,只是想说明基本用法)。

@Test
public void testUpdate() {

    doAnswer(new Answer<Void>() {

        @Override
        public Void answer(InvocationOnMock invocation) throws Throwable {
            Object[] arguments = invocation.getArguments();
            if (arguments != null && arguments.length > 1 && arguments[0] != null && arguments[1] != null) {

                Customer customer = (Customer) arguments[0];
                String email = (String) arguments[1];
                customer.setEmail(email);

            }
            return null;
        }
    }).when(daoMock).updateEmail(any(Customer.class), any(String.class));

    // calling the method under test
    Customer customer = service.changeEmail("old@test.com", "new@test.com");

    //some asserts
    assertThat(customer, is(notNullValue()));
    assertThat(customer.getEmail(), is(equalTo("new@test.com")));

}

@Test(expected = RuntimeException.class)
public void testUpdate_throwsException() {

    doThrow(RuntimeException.class).when(daoMock).updateEmail(any(Customer.class), any(String.class));

    // calling the method under test
    Customer customer = service.changeEmail("old@test.com", "new@test.com");

}
}

你可以在我的文章how to mock with Mockito中找到更多关于如何用Mockito模拟和测试void方法的细节(一个包含示例的全面指南)

看一下Mockito API文档。正如链接文档中提到的(第12点),你可以使用Mockito框架中的doThrow()、doAnswer()、doNothing()、doReturn()方法家族中的任何一个来模拟void方法。

例如,

Mockito.doThrow(new Exception()).when(instance).methodName();

或者如果你想把它和后续行为结合起来,

Mockito.doThrow(new Exception()).doNothing().when(instance).methodName();

假设您正在下面的类World中模拟setter setState(String s),代码使用doAnswer方法来模拟setState。

World mockWorld = mock(World.class); 
doAnswer(new Answer<Void>() {
    public Void answer(InvocationOnMock invocation) {
      Object[] args = invocation.getArguments();
      System.out.println("called with arguments: " + Arrays.toString(args));
      return null;
    }
}).when(mockWorld).setState(anyString());

所谓的问题的解决方案是使用一个间谍Mockito.spy(…)而不是一个mock Mockito.mock(..)。

间谍使我们能够局部嘲笑。Mockito擅长这件事。因为您有一个不完整的类,所以您可以通过这种方式模拟该类中一些必需的位置。

在Java 8中,假设你有一个org.mockito.Mockito.doAnswer的静态导入,这可以做得更清楚一些:

doAnswer(i -> {
  // Do stuff with i.getArguments() here
  return null;
}).when(*mock*).*method*(*methodArguments*);

返回null;是重要的,如果没有它,编译将失败,并出现一些相当模糊的错误,因为它将无法为doAnswer找到合适的覆盖。

例如,ExecutorService可以立即执行传递给execute()的任何可运行对象,可以使用以下方法实现:

doAnswer(i -> {
  ((Runnable) i.getArguments()[0]).run();
  return null;
}).when(executor).execute(any());

加上@sateesh所说的,当你只是想模拟一个void方法以防止测试调用它时,你可以这样使用Spy:

World world = new World();
World spy = Mockito.spy(world);
Mockito.doNothing().when(spy).methodToMock();

当您希望运行测试时,请确保在spy对象上调用in test方法,而不是在world对象上调用。例如:

assertEquals(0, spy.methodToTestThatShouldReturnZero());