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.
Ответы
Ответ 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
- подпись метода остается тем же самым в противном случае