我遇到了一个我认为可能是Mockito的错误,但我想知道是否有其他人能解释为什么这个测试不起作用。
基本上,我有两个对象,像这样:
public class FirstObject {
private SecondObject secondObject;
public SecondObject getSecondObject() { return secondObject; }
}
public class SecondObject {
private String name;
public String getName() { return name; }
}
第一个对象通过注释和前面的方法进行模拟:
@Mock
FirstObject mockedFirstObject;
@Before
public void setup() {
MockitoAnnotations.initMocks(this);
}
第二个对象用一种方法模拟:
public SecondObject setupMockedSecondObject() {
SecondObject secondObject = Mockito.mock(SecondObject.class);
Mockito.when(secondObject.getName()).thenReturn("MockObject");
return secondObject;
}
当thenReturn
包含对该方法的直接调用以设置并获得第二个对象的mock时,它将失败:
@Test
public void notWorkingTest() {
Mockito.when(mockedFirstObject.getSecondObject()).thenReturn(setupMockedSecondObject());
Assert.assertEquals(mockedFirstObject.getSecondObject().getName(), "MockObject");
}
但是,当相同方法返回的mock被分配给thenReturn
中使用的局部变量时,它就起作用了:
@Test
public void workingTest() {
SecondObject mockedSecondObject = setupMockedSecondObject();
Mockito.when(mockedFirstObject.getSecondObject()).thenReturn(mockedSecondObject);
Assert.assertEquals(mockedFirstObject.getSecondObject().getName(), "MockObject");
}
我们是做错了什么,还是这确实是Mockito中的一个错误/限制?这不起作用有什么故意的原因吗?
这确实是Mockito的一个限制,在他们的FAQ:中引用了它
我可以
thenReturn()
还是内联的mock()
不幸的是,你不能这样做:
when(m.foo()).thenReturn(mock(Foo.class)); // ^
原因是,如果我们允许上面的构造,检测未完成的存根将不起作用。我们认为这是对框架验证的"权衡"(另请参阅前面的常见问题条目)。然而,您可以稍微更改代码以使其工作:
//extract local variable and start smiling: Foo foo = mock(Foo.class); when(m.foo()).thenReturn(foo);
如前所述,解决方法是将所需的返回值存储在本地变量中,就像您所做的那样。
我的理解是,Mockito在每次调用它的方法时都会验证你对它的使用。当在正在进行的存根处理过程中调用另一个方法时,您正在破坏其验证过程。
不能在thenReturn
中使用方法,但可以在thenAnswer
中使用您的代码将在条件发生后调用,与基于thenReturn
的任何解决方案不同
这样你就可以写:
@Test
public void nowWorkingTest() {
Mockito.when(mockedFirstObject.getSecondObject()).thenAnswer(new Answer<Map>() {
@Override
public Map answer(InvocationOnMock invocation) {
return setupMockedSecondObject();
}
});
Assert.assertEquals(mockedFirstObject.getSecondObject().getName(), "MockObject");
}
让我们在这里找到另一个例子
@Test
public void testAuthenticate_ValidCredentials() throws FailedToAuthenticateException {
String username = "User1";
String password = "Password";
/*Configure Returning True with when...thenReturn configuration on mock Object - Q5*/
//Write your code here
assertTrue(authenticator.authenticateUser(username, password));
}