Подтвердить что ты не робот

Mockito mock конструктор с параметром

У меня есть класс, как показано ниже:

public class A {
    public A(String test) {
        bla bla bla
    }

    public String check() {
        bla bla bla
    }
}

Логика в конструкторе A(String test) и check() - это вещи, которые я пытаюсь высмеять. Мне нужны любые вызовы вроде: new A($$$any string$$$).check() возвращает фиктивную строку "test".

Я пробовал:

 A a = mock(A.class); 
 when(a.check()).thenReturn("test");

 String test = a.check(); // to this point, everything works. test shows as "tests"

 whenNew(A.class).withArguments(Matchers.anyString()).thenReturn(rk);
 // also tried:
 //whenNew(A.class).withParameterTypes(String.class).withArguments(Matchers.anyString()).thenReturn(rk);

 new A("random string").check();  // this doesn't work

Но он, похоже, не работает. new A($$$any string$$$).check() все еще проходит логику конструктора, а не извлекает издеваемый объект A.

4b9b3361

Ответ 1

Код, который вы опубликовали, работает для меня с последней версией Mockito и Powermockito. Может быть, вы не подготовили A? Попробуйте следующее:

A.java

public class A {
     private final String test;

    public A(String test) {
        this.test = test;
    }

    public String check() {
        return "checked " + this.test;
    }
}

MockA.java

import static org.hamcrest.MatcherAssert.assertThat;
import static org.hamcrest.Matchers.equalTo;
import static org.mockito.Mockito.mock;
import static org.mockito.Mockito.when;

import org.junit.Test;
import org.junit.runner.RunWith;
import org.mockito.Mockito;
import org.powermock.api.mockito.PowerMockito;
import org.powermock.core.classloader.annotations.PrepareForTest;
import org.powermock.modules.junit4.PowerMockRunner;

@RunWith(PowerMockRunner.class)
@PrepareForTest(A.class)
public class MockA {
    @Test
    public void test_not_mocked() throws Throwable {
        assertThat(new A("random string").check(), equalTo("checked random string"));
    }
    @Test
    public void test_mocked() throws Throwable {
         A a = mock(A.class); 
         when(a.check()).thenReturn("test");
         PowerMockito.whenNew(A.class).withArguments(Mockito.anyString()).thenReturn(a);
         assertThat(new A("random string").check(), equalTo("test"));
    }
}

Оба теста должны пройти с mockito 1.9.0, powermockito 1.4.12 и junit 4.8.2

Ответ 2

Насколько я знаю, вы не можете издеваться над конструкторами с mockito, а только с методами. Но согласно вики на кодовой странице Mockito google существует способ издеваться над конструкторским поведением, создав метод в вашем классе, который возвращает новый экземпляр этого класса. то вы можете издеваться над этим методом. Ниже приведен фрагмент http://code.google.com/p/mockito/wiki/MockingObjectCreation

Ответ 3

Без использования Powermock... См. пример ниже, основанный на ответе Бена Глассера, так как мне понадобилось некоторое время, чтобы понять это. Надеюсь, что это экономит несколько раз...

Оригинальный класс:

public class AClazz {

    public void updateObject(CClazz cClazzObj) {
        log.debug("Bundler set.");
        cClazzObj.setBundler(new BClazz(cClazzObj, 10));
    } 
}

Измененный класс:

@Slf4j
public class AClazz {

    public void updateObject(CClazz cClazzObj) {
        log.debug("Bundler set.");
        cClazzObj.setBundler(getBObject(cClazzObj, 10));
    }

    protected BClazz getBObject(CClazz cClazzObj, int i) {
        return new BClazz(cClazzObj, 10);
    }
 }

Класс тестирования

public class AClazzTest {

    @InjectMocks
    @Spy
    private AClazz aClazzObj;

    @Mock
    private CClazz cClazzObj;

    @Mock
    private BClazz bClassObj;

    @Before
    public void setUp() throws Exception {
        Mockito.doReturn(bClassObj)
               .when(aClazzObj)
               .getBObject(Mockito.eq(cClazzObj), Mockito.anyInt());
    }

    @Test
    public void testConfigStrategy() {
        aClazzObj.updateObject(cClazzObj);

        Mockito.verify(cClazzObj, Mockito.times(1)).setBundler(bClassObj);
    }
}

Ответ 4

У Mockito есть ограничения для тестирования конечных, статических и частных методов.

с библиотекой тестирования jMockit, вы можете сделать несколько вещей очень просто и прямолинейно, как показано ниже:

Макет-конструктор класса java.io.File:

new MockUp<File>(){
    @Mock
    public void $init(String pathname){
        System.out.println(pathname);
        // or do whatever you want
    }
};
  • имя публичного конструктора должно быть заменено на $init
  • аргументы и исключенные исключения остаются такими же
  • Тип возврата должен быть определен как void

Откажитесь от статического метода:

  • удалить статическую информацию из метода mock signature
  • подпись метода остается тем же самым в противном случае