it-swarm.com.ru

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

Итак, я создаю фиктивный объект как статическую переменную на уровне класса следующим образом ... В одном тесте я хочу, чтобы 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, что означает, что мне придется выполнять цепочку в каждом тесте.

162
Polaris878

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

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

40
shoebox639

Вы также можете заглушка последовательных вызовов (# 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
    }
}
393
Tony R

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

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

или же

    when(mockFoo.someMethod())
            .thenReturn(obj1, obj2)
            .thenThrow(new RuntimeException("Fail"));
8
Nagy Attila

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

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

doReturn(obj1).doReturn(obj2).when(this.client).someMethod();
5
fl0w

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

when(mockFoo.someMethod()).thenReturn(obj1, obj2);
4
Pedro H