【问题标题】:Mockito: using a method in "thenReturn" to return a mock doesn't workMockito:使用“thenReturn”中的方法返回模拟不起作用
【发布时间】:2016-03-04 11:26:05
【问题描述】:

我遇到了我认为可能是 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");
}

但是,当将相同方法返回的 mock 分配给一个局部变量时,它在 thenReturn 中使用,它可以工作:

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

我们做错了什么还是这确实是 Mockito 中的错误/限制?是否有故意的原因导致这不起作用?

【问题讨论】:

  • 这可能是由于 Mockito 如何实现其when/thenReturn(或其他)调用。在调用前一个 thenReturn 之前,您通过调用另一个 when/thenReturn 循环来打破该链条。

标签: java unit-testing mocking mockito


【解决方案1】:

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

让我们再找一个例子here

【讨论】:

    【解决方案2】:
    @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));
    }
    

    【讨论】:

      【解决方案3】:

      这确实是Mockito的一个限制,引用in their 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 每次调用它的方法时都会验证你对它的使用。在正在进行的存根过程中调用另一个方法时,您将破坏其验证过程。

      【讨论】:

      • 现在可以了!这是一个旧答案。
      • @HamzehSoboh 不,你不能因为这个方法之前运行过,我们希望它按照“then return”的要求执行。这是一个限制,因为在 thenReturn 执行之前,无法通过间谍 foo() 方法更新状态。 thenReturn 参数无法从 spied 或 mocked 方法设置的状态中受益。
      猜你喜欢
      • 1970-01-01
      • 1970-01-01
      • 2023-03-29
      • 2020-01-21
      • 1970-01-01
      • 1970-01-01
      • 1970-01-01
      • 1970-01-01
      • 1970-01-01
      相关资源
      最近更新 更多