使用Mockito测试Callback回调

概述

在这个简短的教程中,我们将重点介绍如何使用流行的测试框架Mockito测试回调。

我们将探索两种解决方案,首先使用ArgumentCaptor,然后使用直观的doAnswer()方法。

Callback回调简介

Callback回调函数是作为参数传递给方法,该方法应在给定时间执行回调参数。

使用回调的一个常见场景是在服务交互期间,当我们需要处理来自服务调用的响应时。比如:

public interface Service {
    void doAction(String request, Callback<Response> callback);
}

在回调参数中,我们传递一个类,该类将使用reply(T response)方法处理响应:

public interface Callback<T> {
    void reply(T response);
}

我们将使用一个简单的服务示例来演示如何传递和调用回调:

public void doAction() {
    service.doAction("our-request", new Callback<Response>() {
        @Override
        public void reply(Response response) {
            handleResponse(response);
        }
    });
}

private void handleResponse(Response response) {
    if (response.isValid()) {
        response.setData(new Data("Successful data response"));
    }
}

我们选择使用Java Lamda表达式写得更简洁:

service.doAction("our-request", response -> handleResponse(response));

使用ArgumentCaptor

现在,让我们看看Mockito如何使用ArgumentCaptor来获取回调对象:

@Test
public void givenServiceWithValidResponse_whenCallbackReceived_thenProcessed() {
    ActionHandler handler = new ActionHandler(service);
    handler.doAction();

    verify(service).doAction(anyString(), callbackCaptor.capture());

    Callback<Response> callback = callbackCaptor.getValue();
    Response response = new Response();
    callback.reply(response);

    String expectedMessage = "Successful data response";
    Data data = response.getData();
    assertEquals(
      "Should receive a successful message: ", 
      expectedMessage, data.getMessage());
}

在本例中,我们首先创建一个ActionHandler,然后调用该处理程序的doAction方法。这只是service.doAction方法调用的包装器,我们在这里调用回调。

接下来,我们验证doAction是否在mock服务实例上被调用,并将anyString()作为第一个参数,将callbackCaptor.capture()作为第二个参数,这就是我们捕获回调对象的地方。然后可以使用getValue()方法返回参数的捕获值。

现在我们有了回调对象,创建一个默认有效的Response对象,直接调用reply方法并断言响应数据具有正确值。

使用doAnswer()方法

现在,我们将使用Mockito的Answer对象和doAnswer方法来存根void方法doAction:

@Test
public void givenServiceWithInvalidResponse_whenCallbackReceived_thenNotProcessed() {
    Response response = new Response();
    response.setIsValid(false);

    doAnswer((Answer<Void>) invocation -> {
        Callback<Response> callback = invocation.getArgument(1);
        callback.reply(response);

        Data data = response.getData();
        assertNull("No data in invalid response: ", data);
        return null;
    }).when(service)
        .doAction(anyString(), any(Callback.class));

    ActionHandler handler = new ActionHandler(service);
    handler.doAction();
}

在第二个示例中,我们首先创建一个无效的Response对象,该对象将在稍后的测试中使用。

接下来,我们在mock服务上设置Answer,以便在调用doAction时,我们拦截调用并使用invocation.getArgument(1)获取方法参数以获取回调参数。

最后一步是创建ActionHandler并调用doAction,从而调用Answer。

结论

本文介绍了在使用Mockito进行测试时处理测试回调的两种不同方法,相比较使用doAnswer方式比较直观。

相关文章

java高级用法之:JNA中的回调

简介什么是callback呢?简单点说callback就是回调通知,当我们需要在某个方法完成之后,或者某个事件触发之后,来通知进行某些特定的任务就需要用到callback了。最有可能看到callbac...

从 Java 程序员到架构师:技术进阶与能力跃迁的完整路径(深度版)

#程序员如何进阶为架构师?#从 Java 程序员到架构师:技术进阶与能力跃迁的完整路径(深度版)一、架构师的核心能力模型架构师是技术与业务的桥梁,需要具备以下核心能力:1.1 系统设计能力案例解析:...

Java 8新特性全面详解:让编程更优雅的“黑科技”集锦

Java 8新特性全面详解:让编程更优雅的“黑科技”集锦1. Lambda表达式:从“命令式”到“声明式”的华丽转身大家还记得那些令人头大的匿名内部类吗?在Java 8之前,实现简单的功能往往需要写一...

JAVA面试之spring篇

(搜索总结)该篇总结的不多,还会继续添加!Spring什么是三级缓存 Spring 使用三级缓存来解决循环依赖问题。三级缓存分别是: 一级缓存(Singleton Objects)1、存储已经完全初始...

Java项目中的异步编程最佳实践

Java项目中的异步编程最佳实践在现代Java开发中,异步编程已经成为提高应用性能和响应能力的重要手段。异步编程的核心在于它允许程序在等待某些操作完成的同时继续处理其他任务,从而充分利用系统资源,提升...