Как имитировать методы void с помощью Mockito
-
21-09-2019 - |
Вопрос
Как имитировать методы с возвращаемым типом void?
Я реализовал шаблон observer, но я не могу издеваться над ним с помощью 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.=( Я хочу показать вышеупомянутое состояние системы.И делайте утверждения в соответствии с ними.
Решение 4
Решение так называемой проблемы заключается в использовании spy
Мокито.шпион(...) вместо того , чтобы mock
Мокито.издеваться(..).
Spy позволяет нам частично издеваться.Мокито хорош в этом вопросе.Поскольку у вас есть класс, который не является полным, таким образом, вы имитируете какое-то требуемое место в этом классе.
Другие советы
Взгляните на Макет Документы API.Как упоминается в связанном документе (пункт № 12), вы можете использовать любой из doThrow()
,doAnswer()
,doNothing()
,doReturn()
семейство методов от фреймворка Mockito до методов mock void.
Например,
Mockito.doThrow(new Exception()).when(instance).methodName();
или, если вы хотите совместить это с последующим поведением,
Mockito.doThrow(new Exception()).doNothing().when(instance).methodName();
Предполагая, что вы смотрите на издевательство над сеттером 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());
Я думаю, что я нашел более простой ответ на этот вопрос, чтобы вызвать реальный метод только для одного метода (даже если он возвращает значение void), вы можете сделать это:
Mockito.doCallRealMethod().when(<objectInstance>).<method>();
<objectInstance>.<method>();
Или вы могли бы вызвать реальный метод для всех методов этого класса, выполнив это:
<Object> <objectInstance> = mock(<Object>.class, Mockito.CALLS_REAL_METHODS);
Добавляя к тому, что сказал @sateesh, когда вы просто хотите смоделировать метод void, чтобы предотвратить его вызов тестом, вы могли бы использовать Spy
вот так:
World world = new World();
World spy = Mockito.spy(world);
Mockito.doNothing().when(spy).methodToMock();
Когда вы хотите запустить свой тест, убедитесь, что вы вызываете метод в test на spy
объект, а не на world
объект.Например:
assertEquals(0,spy.methodToTestThatShouldReturnZero());
Прежде всего:вы всегда должны импортировать mockito static, таким образом, код будет намного более читаемым (и интуитивно понятным).:
import static org.mockito.Mockito.*;
Для частичного издевательства и сохранения оригинальной функциональности в остальных mockito предлагает "Spy".
Вы можете использовать его следующим образом:
private World world = spy(World.class);
Чтобы исключить выполнение метода, вы могли бы использовать что-то вроде этого:
doNothing().when(someObject).someMethod(anyObject());
чтобы придать методу какое-то пользовательское поведение, используйте "when" с "thenReturn".:
doReturn("something").when(this.world).someMethod(anyObject());
Для получения дополнительных примеров, пожалуйста, найдите образцы mockito в документе.
Как имитировать методы void с помощью mockito - есть два варианта:
doAnswer
- Если мы хотим, чтобы наш издевательский метод void что-то делал (издевался над поведением, несмотря на то, что оно недействительно).doThrow
- Тогда естьMockito.doThrow()
если вы хотите создать исключение из издевательского метода 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");
}
}
Вы могли бы найти более подробную информацию о том, как издеваться и тест пустота методы с Mockito в моем посте Как издеваться с помощью Mockito (Подробное руководство с примерами)
Добавляем еще один ответ к этой связке (без каламбура)...
Вам действительно нужно вызвать метод doAnswer, если вы не можете \ не хотите использовать spy.Однако вам не обязательно сворачивать свой собственный Ответ.Существует несколько реализаций по умолчанию.Примечательно, Вызывает реальные методы.
На практике это выглядит примерно так:
doAnswer(new CallsRealMethods()).when(mock)
.voidMethod(any(SomeParamClass.class));
Или:
doAnswer(Answers.CALLS_REAL_METHODS.get()).when(mock)
.voidMethod(any(SomeParamClass.class));
В Java 8 это можно сделать немного чище, предполагая, что у вас есть статический импорт для org.mockito.Mockito.doAnswer
:
doAnswer(i -> {
// Do stuff with i.getArguments() here
return null;
}).when(*mock*).*method*(*methodArguments*);
В return null;
это важно, и без этого компиляция завершится неудачей с некоторыми довольно неясными ошибками, поскольку она не сможет найти подходящее переопределение для doAnswer
.
Например , an ExecutorService
это просто немедленно выполняет любой Runnable
переданный в execute()
может быть реализован с использованием:
doAnswer(i -> {
((Runnable) i.getArguments()[0]).run();
return null;
}).when(executor).execute(any());
Я думаю, что ваши проблемы связаны с вашей структурой теста.Мне было трудно смешивать mocking с традиционным методом реализации интерфейсов в тестовом классе (как вы сделали здесь).
Если вы реализуете прослушиватель как макет, вы можете затем проверить взаимодействие.
Listener listener = mock(Listener.class);
w.addListener(listener);
world.doAction(..);
verify(listener).doAction();
Это должно убедить вас в том, что "Мир" поступает правильно.