我在编写涉及模拟lambda的单元测试时遇到了一个有趣的问题。
@Test
public void spyingLambda() {
final Supplier<String> spy = Mockito.spy((Supplier) () -> "1");
spy.get();
}
运行此测试失败,并显示以下错误:
莫基托不能嘲笑/间谍,因为: - 期末班
上述问题的一种解决方法是将 lambda 替换为匿名实现:
@Test
public void spyingAnonymousImplementation() {
final Supplier<String> spy = Mockito.spy(new Supplier<String>() {
@Override
public String get() {
return "1";
}
});
spy.get();
}
尽管这两个测试完全相同(IDE 建议甚至用 lambda 替换匿名实现),但第二个测试不会失败。
我想知道这是否是一个可以在 mockito 中修复的已知问题,或者是否有任何其他解决方法。
处理此问题的另一种方法如下:
/**
* This method overcomes the issue with the original Mockito.spy when passing a lambda which fails with an error
* saying that the passed class is final.
*/
@SuppressWarnings("unchecked")
static <T, P extends T> P spyLambda(Class<T> lambdaType, P lambda) {
return (P) mock(lambdaType, delegatesTo(lambda));
}
这允许通过更改第一种方法来监视 lambda,如下所示:
@Test
void spyingLambda() {
Supplier<String> spy = spyLambda(Supplier.class, () -> "1");
spy.get();
}
希望上述示例可以帮助遇到相同问题的其他人。
仅供参考,为了提高@alex的答案,你也可以做
public static <T> T spyLambda(final T lambda) {
Class<?>[] interfaces = lambda.getClass().getInterfaces();
MatcherAssert.assertThat(interfaces, IsArrayWithSize.arrayWithSize(1));
return Mockito.mock((Class<T>) interfaces[0], delegatesTo(lambda));
}
然后简单地监视它而不指定类型(例如,Supplier.class
Callable<Integer> callable = spyLambda(() -> {
return 42;
});
Supplier<Integer> supplier = spyLambda(() -> 42);
Runnable runnable = spyLambda(() -> System.out.println("42"));
您可以允许最终类模拟。创建包含以下内容的文件src/test/resources/mockito-extensions/org.mockito.plugins.MockMaker
mock-maker-inline
https://www.baeldung.com/mockito-final#configure-mocktio