JUnit 使用 Mockito 测试异步方法

Gar*_*ead 5 java junit spring mockito spring-scheduled

我已经使用 Spring Framework(版本 5.0.5.RELEASE)在 Java 1.8 类中实现了一个异步方法:

public class ClassToBeTested {
    @Autowired
    private MyComponent myComponent;

    @Async
    public void doStuff(List<MyClass> myObjects) {
        CompletableFuture<MyResponseObject>[] futureList = new CompletableFuture[myObjects.size()];
        int count = 0;

        for (MyClass myObject : myObjects) {
            futureList[count] = myComponent.doOtherStuff(myObject);
            count++;
        }

        // Wait until all doOtherStuff() calls have been completed
        CompletableFuture.allOf(futureList).join();

        ... other stuff ...
    }
}
Run Code Online (Sandbox Code Playgroud)

我正在尝试使用 JUnit 和 Mockito 测试该类。我已将其设置如下,目的是模拟 doStuff() 方法对组件的调用:

@MockBean
private MyComponent myComponentAsAMock;

@InjectMocks
@Autowired
private ClassToBeTested classToBeTested;

@Test
public void myTest() throws Exception {
    // Create object to return when myComponent.doOtherStuff() is called.
    CompletableFuture<MyResponseObject> completableFuture = new CompletableFuture<MyResponseObject>();
    ... populate an instance of MyResponseObject ...
    completableFuture.complete(myResponseObject);

    // Return object when myComponent.doOtherStuff() is called.
    Mockito.when(
        myComponentAsAMock.doOtherStuff(ArgumentMatchers.any(MyClass.class)))
        .thenReturn(completableFuture);

    // Test.
    List<MyClass> myObjects = new ArrayList<MyClass>();
    MyClass myObject = new MyClass();
    myobjects.add(myObject);
    classToBeTested.doStuff(myObjects);
}
Run Code Online (Sandbox Code Playgroud)

虽然当我在 Eclipse 中单独运行单元测试时似乎是成功的,但是对整个项目进行 Maven 构建时,我注意到正在抛出 NullPointerExceptions:

[ThreadExecutor2] .a.i.SimpleAsyncUncaughtExceptionHandler : Unexpected error occurred invoking async method 'public void package.ClassToBeTested.doStuff(java.util.List)'.

java.lang.NullPointerException: null
at java.util.concurrent.CompletableFuture.andTree(CompletableFuture.java:1306) ~[na:1.8.0_131]
at java.util.concurrent.CompletableFuture.allOf(CompletableFuture.java:2225) ~[na:1.8.0_131]
at package.ClassToBeTested.doStuff(ClassToBeTested.java:75) ~[classes/:na]
Run Code Online (Sandbox Code Playgroud)

在 ClassToBeTested.java 的这一行上引发了错误:

CompletableFuture.allOf(completedFutureList).join();
Run Code Online (Sandbox Code Playgroud)

看起来像测试完成后,显示在Maven构建输出异常消息(还有其他的测试正在运行,其输出显示错误消息之前发生),所以我猜测它的东西与事实对 doStuff() 的调用是异步的。

任何援助将不胜感激。

Gar*_*ead 17

解决方案是添加一个带有超时和检查的 Mockito 验证步骤,以确保模拟组件的方法已被调用适当的次数:

    Mockito.verify(myComponentAsAMock, Mockito.timeout(1000).times(1)).doOtherStuff(ArgumentMatchers.any(MyClass.class));
Run Code Online (Sandbox Code Playgroud)