4

我用对象列表调用方法 performAction 并验证相同。调用此方法后,我修改了一些“对象”。

Mockito 验证失败,说参数不匹配(显示修改后的对象),但我可以在调试模式下看到对象根据需要是正确的。

理想情况下,这不应该发生,因为应该根据实际调用方法的时间来应用验证。在测试方法中的验证调用期间是否比在模拟方法调用时应用验证?

测试班

@Test
public void test() throws Exception {
    List<ABC> objects = new ArrayList<ABC>();
    //populate objects.
    activity.performActions(objects);               
    verify(activity, times(1)).doActivity(objects);
}

测试方法:

public void performActions(List<ABC> objects) {

    activity.doActivity(urlObjects2PerformAction);
    //Modify objects                

}

我得到的错误如下(这是完整的代码。我给出了最小的代码段):

Argument(s) are different! Wanted:
activity.doActivity(
.......
......
4

3 回答 3

8

以前有人问过这个问题-在Mockito 可以根据方法调用时的值验证参数吗?

当您调用已使用 Mockito 存根的方法时,Mockito 将存储传递给它的参数,以便您verify以后使用。也就是说,它存储对象引用,而不是对象本身的内容。如果您稍后更改这些对象的内容,那么您的verify调用会将其参数与更新的对象进行比较 - 它不会对原始对象进行深层复制。

如果您需要验证对象的内容是什么,则需要

  • 在方法调用时自己存储它们;或者
  • 在方法调用时验证它们。

执行其中任何一项的正确方法是使用 Mockito Answer。因此,对于第二个选项,您将创建一个Answer进行验证,并AssertionFailedError在参数值不正确时抛出一个;而不是verify在测试结束时使用。

于 2013-08-24T22:17:45.327 回答
1

verify比较调用时的参数内容,verify而不是调用模拟方法时的参数内容。如果列表的内容被修改,那么verify将使用修改后的值。

另一种方法是在调用方法后立即使用 anAnswer来检查参数,或者您可以创建一个新列表而不是修改旧列表。

于 2013-08-24T21:46:23.420 回答
0

现在可以使用 ArgumentCaptor 解决这个问题

@Test
public void test() throws Exception {
    List<ABC> objects = new ArrayList<ABC>();
    ArgumentCaptor<List<ABC> objectsCaptor = ArgumentCaptor<List.class>;
    //populate objects.
    activity.performActions(objects);               
    verify(activity, times(1)).doActivity(objectsCaptor.capture());
}
于 2018-11-12T22:40:06.630 回答