Как сделать mock to void методами с Mockito - PullRequest
820 голосов
/ 16 февраля 2010

Как смоделировать методы с возвращаемым типом 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();
}

Система не запускается с макетом. знак равно Я хочу показать вышеупомянутое состояние системы. И сделайте утверждение в соответствии с ними.

Ответы [ 10 ]

1033 голосов
/ 17 февраля 2010

Взгляните на документацию по Mockito API . Как упоминается в связанном документе (Точка # 12), вы можете использовать любой из методов семейства doThrow(), doAnswer(), doNothing(), doReturn() из среды Mockito для моделирования методов void.

Например,

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

или, если вы хотите объединить это с последующим поведением,

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

Предполагая, что вы смотрите на издевательство над сеттером setState(String s) в классе World, приведенном ниже, код использует 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());
94 голосов
/ 11 июня 2014

Мне кажется, я нашел более простой ответ на этот вопрос: вызвать реальный метод только для одного метода (даже если он имеет возврат void), вы можете сделать это:

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

Или вы можете вызвать реальный метод для всех методов этого класса, сделав это:

<Object> <objectInstance> = mock(<Object>.class, Mockito.CALLS_REAL_METHODS);
62 голосов
/ 09 января 2015

В дополнение к тому, что сказал @sateesh, когда вы просто хотите смоделировать метод void, чтобы предотвратить вызов его тестом, вы можете использовать Spy следующим образом:

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

Если вы хотите запустить тест, убедитесь, что вы вызываете тестируемый метод для объекта spy, а не для объекта world. Например:

assertEquals(0,spy.methodToTestThatShouldReturnZero());
52 голосов
/ 18 февраля 2010

Решением так называемой проблемы является использование spy Mockito.spy (...) вместо mock Mockito.mock (..),

Шпион позволяет нам частичное издевательство. Мокито хорош в этом вопросе. Поскольку у вас есть незавершенный класс, таким образом вы высмеиваете какое-то необходимое место в этом классе.

24 голосов
/ 11 ноября 2016

Прежде всего: вы всегда должны импортировать mockito static, таким образом, код будет намного более читабельным (и интуитивно понятным):

import static org.mockito.Mockito.*;

Для частичной насмешки и при сохранении оригинальной функциональности в остальном mockito предлагает «Шпион».

Вы можете использовать его следующим образом:

private World world = spy(World.class);

Чтобы исключить выполнение метода, вы можете использовать что-то вроде этого:

doNothing().when(someObject).someMethod(anyObject());

для придания некоторого пользовательского поведения методу, использующему «when» с «thenReturn»:

doReturn("something").when(this.world).someMethod(anyObject());

Для большего количества примеров, пожалуйста, найдите образцы мокито в документе.

19 голосов
/ 25 мая 2017

Как смоделировать пустые методы с помощью mockito - есть два варианта:

  1. doAnswer - Если мы хотим, чтобы наш метод mocked void что-то делал (смоделируйте поведение, несмотря на то, что void).
  2. doThrow - Тогда есть Mockito.doThrow(), если вы хотите создать исключение из метода mocked void.

Ниже приведен пример того, как его использовать (не идеальный вариант использования, а просто хотел проиллюстрировать основное использование).

@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");

}
}

Более подробную информацию о том, как mock и test void методами с Mockito, можно найти в моем посте Как издеваться над Mockito руководство с примерами)

14 голосов
/ 19 марта 2014

Добавление еще одного ответа в группу (без каламбура) ...

Вам необходимо вызвать метод doAnswer, если вы не можете \ не хотите использовать шпионы. Тем не менее, вам не обязательно бросать свой ответ . Есть несколько реализаций по умолчанию. В частности, CallsRealMethods .

На практике это выглядит примерно так:

doAnswer(new CallsRealMethods()).when(mock)
        .voidMethod(any(SomeParamClass.class));

Или:

doAnswer(Answers.CALLS_REAL_METHODS.get()).when(mock)
        .voidMethod(any(SomeParamClass.class));
8 голосов
/ 11 декабря 2017

В Java 8 это можно сделать немного чище, при условии, что у вас есть статический импорт для org.mockito.Mockito.doAnswer:

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

Значение return null; важно, и без него компиляция завершится неудачно с некоторыми довольнонеясные ошибки, так как он не сможет найти подходящее переопределение для doAnswer.

Например, ExecutorService, который просто немедленно выполняет любой Runnable, переданный execute(), может быть реализован с использованием:

doAnswer(i -> {
  ((Runnable) i.getArguments()[0]).run();
  return null;
}).when(executor).execute(any());
5 голосов
/ 07 июля 2014

Я думаю, что ваши проблемы связаны с вашей тестовой структурой. Мне было трудно смешивать макеты с традиционным методом реализации интерфейсов в тестовом классе (как вы сделали здесь).

Если вы реализуете слушателя как Mock, вы можете проверить взаимодействие.

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

Это должно убедить вас, что «Мир» делает правильные вещи.

0 голосов
/ 14 февраля 2017

@ Эшли: у меня работает

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);
} }
...