0

我正在使用 mockito 来测试一个通用方法。但是当我运行 junit-test 时,我得到了一个 ClassCastException。

被测方法如下所示:

public ExampleClass {
    public <E> E randomObject(List<E> list) {
            E e = list.get(0);
            return e;
    }
}

模拟看起来像这样:

ExampleClass exampleMock = mock(ExampleClass.class);
List listMock = mock(List.class);
when(exampleMock.randomObject(Matchers<List<String>any())).thenReturn("Hello");
when(exampleMock.randomObject(Matchers.List<Integer>any())).thenReturn(20);

异常出现在when-method的第二个定义中。看起来该方法只接受一种不可更改的类型。但为什么会这样呢?当我在普通 java 中使用具有两种不同类型的泛型方法时,不会出现异常。

有人可以帮忙吗?

4

1 回答 1

2

tl;博士

将代码拆分为多种测试方法。或者链接thenReturnAPI。

很长的故事

您可能知道 Java 泛型是通过类型擦除实现的,这意味着您在代码中看到的大多数泛型仅存在于源代码中,而不存在于编译后的字节码中。

例如这里的以下签名

<E> E randomObject(List<E> list)

编译为

Object randomObject(List list)

这就是 mockito 看到的签名。matchers 也是如此:

when(exampleMock.randomObject(Matchers.<List<String>>any())).thenReturn("Hello");
when(exampleMock.randomObject(Matchers.<List<Integer>>any())).thenReturn(20);

变成

when(exampleMock.randomObject(Matchers.any())).thenReturn("Hello");
when(exampleMock.randomObject(Matchers.any())).thenReturn(20);

Mockito 发现这是同一个 matcher同一个调用。虽然它的代码读起来像两个不同的存根,但 mockito 不知道这一点,只能假设开发人员想要覆盖第一个存根。

对于在某些方法中声明默认存根@Before并且需要在某些方法中覆盖该存根的场景,需要这种模拟行为@Test

在这种情况下,代码应该被拆分,或者如果是更复杂场景的一部分,存根应该使用链 API,例如

when(exampleMock.randomObject(Matchers.anyList()))
       .thenReturn("Hello")
       .thenReturn(20);

_请注意,匹配器已更改为anyList()可能对编译时间更友好的匹配器。

测试可以用 注释@SuppressWarnings("unchecked")

于 2015-03-01T23:33:32.920 回答