[java] 如何使用mockito模擬void方法



4 Answers

我想我已經找到了一個更簡單的答案來回答這個問題,為一個方法調用真正的方法(即使它有一個無效返回),你可以這樣做:

Mockito.doCallRealMethod().when(<objectInstance>).<method>();
<objectInstance>.<method>();

或者,你可以為這個類的所有方法調用真正的方法,這樣做:

<Object> <objectInstance> = mock(<Object>.class, Mockito.CALLS_REAL_METHODS);
Question

如何用void返回類型來模擬方法?

我實現了觀察者模式,但我不能用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();
}

系統不會被模擬觸發。 =(我想顯示上面提到的系統狀態,並根據它們進行斷言。




所謂問題的解決方案是使用spy Mockito.spy(...)而不是mock Mockito.mock(..)

間諜使我們能夠部分嘲笑。 Mockito擅長這件事。 因為你有不完整的課程,所以你在這個課堂上嘲笑了一些必要的地方。




如何用mockito模擬無效方法 - 有兩種選擇:

  1. doAnswer - 如果我們想要我們的虛擬無效方法來做一些事情(儘管是無效的嘲笑行為)。
  2. doThrow - 如果你想從Mockito.doThrow()的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");

}
}

你可以在我的文章中找到更多關於如何用Mockito 模擬測試 void方法的細節。如何用Mockito 模擬(一個帶有例子的綜合指南)




我認為你的問題是由於你的測試結構。 我發現很難將嘲諷與在測試類中實現接口的傳統方法混淆(正如您在此處所做的那樣)。

如果您將偵聽器實現為模擬,則可以驗證交互。

Listener listener = mock(Listener.class);
w.addListener(listener);
world.doAction(..);
verify(listener).doAction();

這應該讓你滿意,'世界'正在做正確的事情。




@ashley:適合我

public class AssetChangeListenerImpl extends
AbstractAssetChangeListener implements AssetChangeListener {

  @Override
  public void onChangeEvent(final EventMessage message) throws EventHubClientException {
      execute(message);
    }
  }
}

public class AbstractAssetChangeListener {
  protected  void execute( final EventMessage message ) throws EventHubClientException {
    executor.execute( new PublishTask(getClient(), message) );
} } @RunWith(MockitoJUnitRunner.class) public class AssetChangeListenerTest extends AbstractAssetChangeListenerTest {

 public void testExecute() throws EventHubClientException {
    EventMessage message = createEventMesage(EventType.CREATE);
    assetChangeListener.execute(message);
    verify(assetChangeListener, times(1)).execute(message);
} }



Related