Как сказать, чтобы объект Mockito возвращал что-то другое при следующем вызове? - PullRequest
162 голосов
/ 18 ноября 2010

Итак, я создаю фиктивный объект как статическую переменную на уровне класса следующим образом ... В одном тесте я хочу, чтобы Foo.someMethod() возвращал определенное значение, а в другом тесте я хочу, чтобы он возвращал другое значение. Проблема в том, что мне кажется, что мне нужно перестроить макеты, чтобы заставить это работать правильно. Я хотел бы избежать перестройки макетов и просто использовать одни и те же объекты в каждом тесте.

class TestClass {

    private static Foo mockFoo;

    @BeforeClass
    public static void setUp() {
        mockFoo = mock(Foo.class);
    }

    @Test
    public void test1() {
        when(mockFoo.someMethod()).thenReturn(0);

        TestObject testObj = new TestObject(mockFoo);

        testObj.bar(); // calls mockFoo.someMethod(), receiving 0 as the value

    }

    @Test
    public void test2() {
        when(mockFoo.someMethod()).thenReturn(1);

        TestObject testObj = new TestObject(mockFoo);

        testObj.bar(); // calls mockFoo.someMethod(), STILL receiving 0 as the value, instead of expected 1.

    }

}

Во втором тесте я все еще получаю 0 в качестве значения при вызове testObj.bar () ... Каков наилучший способ решить эту проблему? Обратите внимание, что я знаю, что могу использовать разные макеты Foo в каждом тесте, однако мне нужно объединить несколько запросов от mockFoo, что означает, что мне придется выполнять цепочку в каждом тесте.

Ответы [ 5 ]

393 голосов
/ 20 мая 2011

Вы также можете Заглушка последовательных вызовов (# 10 в 2.8.9 api). В этом случае вы будете использовать несколько вызовов thenReturn или один вызов thenReturn с несколькими параметрами (varargs).

import static org.junit.Assert.assertEquals;
import static org.mockito.Mockito.mock;
import static org.mockito.Mockito.when;

import org.junit.Before;
import org.junit.Test;

public class TestClass {

    private Foo mockFoo;

    @Before
    public void setup() {
        setupFoo();
    }

    @Test
    public void testFoo() {
        TestObject testObj = new TestObject(mockFoo);

        assertEquals(0, testObj.bar());
        assertEquals(1, testObj.bar());
        assertEquals(-1, testObj.bar());
        assertEquals(-1, testObj.bar());
    }

    private void setupFoo() {
        mockFoo = mock(Foo.class);

        when(mockFoo.someMethod())
            .thenReturn(0)
            .thenReturn(1)
            .thenReturn(-1); //any subsequent call will return -1

        // Or a bit shorter with varargs:
        when(mockFoo.someMethod())
            .thenReturn(0, 1, -1); //any subsequent call will return -1
    }
}
40 голосов
/ 18 ноября 2010

Прежде всего, не делайте макет статичным. Сделайте это частным полем. Просто поместите ваш класс setUp в @Before, а не @BeforeClass. Это может быть много, но это дешево.

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

9 голосов
/ 24 мая 2018

Для всех, кто ищет что-то для возврата, а затем для другого исключения броска вызова:

    when(mockFoo.someMethod())
            .thenReturn(obj1)
            .thenReturn(obj2)
            .thenThrow(new RuntimeException("Fail"));

или

    when(mockFoo.someMethod())
            .thenReturn(obj1, obj2)
            .thenThrow(new RuntimeException("Fail"));
6 голосов
/ 07 марта 2018

Для тех, кто использует spy () и doReturn () вместо метода when ():

то, что вам нужно, чтобы вернуть другой объект при разных вызовах, это:

doReturn(obj1).doReturn(obj2).when(this.client).someMethod();
5 голосов
/ 24 октября 2018

Или даже чище:

when(mockFoo.someMethod()).thenReturn(obj1, obj2);
...