49

我正在使用模拟 1.9.5。我有以下代码:

public class ClassA  {

public List<? extends MyInterface> getMyInterfaces() {
    return null;
}

public static void testMock() {
    List<MyInterface> interfaces = new ArrayList<>();
    ClassA classAMock = mock(ClassA.class);
    when(classAMock.getMyInterfaces()).thenReturn(interfaces);      
}

我得到一个编译错误的thenReturn(interfaces)说法:

"The method thenReturn(List<capture#1-of ? extends MyInterface>) in the type 
 OngoingStubbing<List<capture#1-of ? extends MyInterface>> is not applicable for the arguments 
 (List<MyInterface>)"

但是,当我使用thenAnswermockito 的方法时,我没有得到错误。谁能告诉我发生了什么事?为什么我在使用该thenReturn方法时会收到错误消息?ClassA当由第 3 方提供且无法修改时,还有其他方法可以解决此问题吗?

4

2 回答 2

65

编辑:从 Mockito 1.10.x 开始,嵌入类中的泛型类型现在由 Mockito 用于深度存根。IE。

public interface A<T extends Observer & Comparable<? super T>>  {
  List<? extends B> bList();
  T observer();
}

B b = deep_stubbed.bList().iterator().next(); // returns a mock of B ; mockito remebers that A returns a List of B
Observer o = deep_stubbed.observer(); // mockito can find that T super type is Observer
Comparable<? super T> c = deep_stubbed.observer(); // or that T implements Comparable

Mockito 尽力获取编译器嵌入的类型信息,但是当应用擦除时,mockito 只能返回一个Object.


原文:嗯,与 Mockito 相比,这更多是泛型的问题。对于泛型,您应该阅读Angelika Langer在它们上写的内容。而对于当前主题,即通配符,请阅读本

但简而言之,您可以使用 Mockito 的其他语法来帮助您解决当前情况:

doReturn(interfaces).when(classAMock).getMyInterfaces();

或者使用 BDD 别名:

willReturn(interfaces).given(classAMock).getMyInterfaces();

不过,您可以编写更通用友好的包装器。这将有助于未来的开发人员使用相同的 3rd 方 API。


作为旁注:你不应该模拟你不拥有的类型,它会导致许多错误和问题。相反,你应该有一些包装。例如 DAO 和存储库就代表了这样的想法,人们将模拟 DAO 或存储库接口,而不是 JDBC / JPA / hibernate 的东西。有很多关于此的博客文章:

于 2013-04-11T09:16:38.580 回答
56

另一种解决方案(尽管可读性较差)是限定静态方法调用when来绑定通配符:

Mockito.<List<? extends MyInterface>>when(classAMock.getMyInterfaces()).thenReturn(interfaces);
于 2013-10-01T14:44:25.117 回答