Mockito:使用“thenReturn”中的方法返回模拟不起作用

时间:2015-11-30 18:10:34

标签: java unit-testing mocking mockito

我遇到了我认为可能是Mockito的错误,但是想知道是否有其他人可以解释为什么这个测试不起作用。

基本上,我有两个对象,如:

public class FirstObject {
    private SecondObject secondObject;
    public SecondObject getSecondObject() { return secondObject; }
}

public class SecondObject {
    private String name;
    public String getName() { return name; }
}

通过注释和before方法模拟第一个对象:

@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包含对此方法的直接调用以设置并获取第二个对象的模拟时,它会失败:

@Test
public void notWorkingTest() {
    Mockito.when(mockedFirstObject.getSecondObject()).thenReturn(setupMockedSecondObject());
    Assert.assertEquals(mockedFirstObject.getSecondObject().getName(), "MockObject");
}

但是,当同一方法返回的模拟被分配给thenReturn中使用的局部变量时,它可以工作:

@Test
public void workingTest() {
    SecondObject mockedSecondObject = setupMockedSecondObject();
    Mockito.when(mockedFirstObject.getSecondObject()).thenReturn(mockedSecondObject);
    Assert.assertEquals(mockedFirstObject.getSecondObject().getName(), "MockObject");
}

我们做错了什么,或者这确实是Mockito的错误/限制?有没有故意这样做不起作用?

3 个答案:

答案 0 :(得分:37)

这确实是Mockito的限制,它被引用in their FAQ

  

我可以thenReturn()内联mock()吗?

     

不幸的是你不能这样做:

when(m.foo()).thenReturn(mock(Foo.class));
//                         ^
     

原因是如果我们允许上述构造,检测未完成的存根将不起作用。我们认为这是一种“权衡”。框架验证(参见前面的FAQ条目)。但是,您可以稍微更改代码以使其正常工作:

//extract local variable and start smiling:
Foo foo = mock(Foo.class);
when(m.foo()).thenReturn(foo);

如上所述,解决方法是将所需的返回值存储在局部变量中,就像您所做的那样。

我理解的方式是,每当你调用它的方法时,Mockito会验证你使用它的用法。在正在进行的存根过程中调用另一个方法时,您将破坏其验证过程。

答案 1 :(得分:1)

您不能在 thenReturn 中使用方法,但可以在 thenAnswer 中使用 您的代码将在 when 条件发生后被调用, 与任何基于 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");
}

再找一个例子here

答案 2 :(得分:0)

@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));
}