4

我对 Mockito 有疑问。

有没有可能做这样的事情:

ClassX x = mock(ClassX.class)
when(x.methodB()).thenReturn("toto");
String result = x.methodA();

我正在使用 Mockito 1.7。

我看到有一个“间谍”系统,但他们说不建议在我们测试的项目上使用它(为什么?)......

无论如何,我尝试了那个间谍功能,但我得到了一个奇怪的行为。

检查我想要做什么:

真实代码:

String methodA(String arg) {
    return this.methodB(arg);
}

String methodB(String arg) {
    return "toto";
}

测试代码:

@Test
public void testTest() {
    final ClassX x = spy( new ClassX() );
    final String argument = "arg";
    doReturn("good").when(helper).methodB(argument);
    assertTrue(  x.methodB(argument).equals("good") );
    assertTrue(  x.methodA(argument).equals("good") );
}  

正如他们所说,我避免了可能是间谍问题的 when thenReturn 语法(但无论如何它也不起作用)

奇怪的是: assertTrue( x.methodB(argument).equals("good") ); 没问题

只有第二个 assertTrue( x.methodA(argument).equals("good") ); 不行

实际上 helper.methodA(argument) 返回 "toto" -> 真实结果而不是模拟结果

在这种情况下,不可能告诉 mockito 返回“好”???似乎当测试类调用methodB时没问题,但是如果间谍的方法调用methodB它就不再起作用了......

我不知道该怎么办......对同一个类的2个方法进行单元测试并让测试彼此独立以使一个著名的模拟测试框架无法实现这个基本功能是不是很奇怪? ? 这不就是我们所说的真正的单元测试吗?不明白为什么他们说要避免对被测对象使用 spy 方法...

谢谢

4

2 回答 2

4

更新:我写了下面的东西,然后过了一会儿发现 .thenCallRealMethod() 可以让你有效地执行部分存根。Mockito 作者建议您使用重构将依赖项分离到不同的类中;但它们确实提供了部分存根的方法。我添加了一个测试方法来演示这种方法,并留下我原来的评论。

原创:我真的很喜欢 Mockito,但这是 EasyMock 胜出的地方。我有两个不涉及 Mockito 的解决方案。第一个是在您的测试实例上覆盖 methodB。另一种是使用 EasyMock 进行部分模拟:

import org.junit.Test;
import static org.junit.Assert.*;
import static org.easymock.EasyMock.*;

public class PartialMockTest {

    class ClassX {
        String methodA(String arg) {return methodB(arg);}
        String methodB(String arg) {return "toto";}
    }

    @Test
    public void MockitoOnClassX(){
        ClassX classx = mock(ClassX.class);
        when(classx.methodB("hiyas")).thenReturn("tomtom");
        when(classx.methodA(anyString())).thenCallRealMethod();
        String response = classx.methodA("hiyas");
        assertEquals("tomtom",response);
    }


    @Test
    public void OverrideOnClassX() {
        ClassX classx = new ClassX(){@Override String methodB(String arg){return "tomtom";}};
        String response = classx.methodA("hiyas");
        assertEquals("tomtom",response);
    }

    @Test
    public void PartialMockOnClassX() throws NoSuchMethodException {
        ClassX classx = createMockBuilder(ClassX.class).addMockedMethod("methodB").createMock();
        expect(classx.methodA("hiyas")).andReturn("tomtom");
        replay(classx);
        String response = classx.methodA("hiyas");
        assertEquals("tomtom",response);
    }

}
于 2011-01-03T20:39:17.440 回答
2

spy 是与被监视对象不同的对象。spy 只委托给被监视的对象。所以当被窥探对象从methodA调用methodB时,它会调用它自己,而不是spy。

于 2010-11-22T14:08:50.280 回答