0

0

使用 Mockito 验证 Executor 内部方法调用

碧海醫心

碧海醫心

发布时间:2025-08-31 16:26:01

|

494人浏览过

|

来源于php中文网

原创

使用 mockito 验证 executor 内部方法调用

本文将探讨在使用 Mockito 框架进行单元测试时,如何验证在 Executor.execute() 方法内部调用的方法。 当被验证的方法调用发生在由 Executor 执行的异步任务中时,直接使用 Mockito.verify() 可能会导致验证失败,因为验证线程和实际执行线程不同。 解决此问题的一种有效方法是使用同步执行器,例如 SynchronousExecutor,它可以强制任务同步执行,从而简化测试并确保方法调用可以被正确验证。

问题背景

在单元测试中,我们经常需要验证某个方法是否被调用以及调用了多少次。当方法调用发生在异步执行的环境中,例如通过 Executor 框架提交的任务,验证过程会变得复杂。 默认情况下,Executor 会在不同的线程中执行任务,这导致 Mockito 的验证机制无法直接捕获到方法调用。

解决方案:使用 SynchronousExecutor

SynchronousExecutor 是一个简单的 Executor 实现,它会在调用线程中立即执行提交的任务,而不是将其提交到单独的线程。 这使得我们可以像同步代码一样验证方法调用。

步骤如下:

  1. 创建 SynchronousExecutor 实例:

    Executor executor = new SynchronousExecutor();
  2. 将 SynchronousExecutor 注入到被测试类中:

    这一步至关重要,确保被测试类使用我们控制的 Executor 实例。 具体注入方式取决于被测试类的设计,可以使用构造函数注入、setter 方法注入或字段注入。 例如,如果被测试类如下:

    兴泰网络办公系统1.61
    兴泰网络办公系统1.61

    基于 Internet 的 Web 技术,完全采用B/S 体系结构的网络办公系统。该系统具有安全性高、功能极为强大、可在广域网中使用也可在局域网中使用、也可以同时在局域网和广域网中使用的特点,全傻瓜式安装,无需作复杂配置,界面采用类似windows资源管理器的设计,结构清晰,条理分明,即使不熟悉电脑的人也可很快掌握全部操作。该系统通过在广域网内的广泛试用验证和经专业技术人员的调试、测试,确认具有很

    下载
    public class MyClass {
        private final Executor executor;
        private final MessageHandler messageHandler;
    
        public MyClass(Executor executor, MessageHandler messageHandler) {
            this.executor = executor;
            this.messageHandler = messageHandler;
        }
    
        public void doSomething(Message message) {
            executor.execute(() -> prepareContext(message));
        }
    
        private void prepareContext(Message message) {
            messageHandler.handleMessage(message);
        }
    }

    那么在测试中,我们需要这样创建 MyClass 实例:

    MessageHandler messageHandler = Mockito.mock(MessageHandler.class);
    Executor executor = new SynchronousExecutor();
    MyClass myClass = new MyClass(executor, messageHandler);
  3. 执行被测试方法:

    Message message = new Message(); // 假设 Message 是一个类
    myClass.doSomething(message);
  4. 验证方法调用:

    Mockito.verify(messageHandler).handleMessage(message);

示例代码

以下是一个完整的示例,展示如何使用 SynchronousExecutor 验证 handleMessage 方法的调用:

import org.junit.jupiter.api.Test;
import org.mockito.Mockito;

import java.util.concurrent.Executor;

import static org.mockito.Mockito.verify;

class MyClassTest {

    interface MessageHandler {
        void handleMessage(Message message);
    }

    static class Message {}

    static class MyClass {
        private final Executor executor;
        private final MessageHandler messageHandler;

        public MyClass(Executor executor, MessageHandler messageHandler) {
            this.executor = executor;
            this.messageHandler = messageHandler;
        }

        public void doSomething(Message message) {
            executor.execute(() -> prepareContext(message));
        }

        private void prepareContext(Message message) {
            messageHandler.handleMessage(message);
        }
    }

    static class SynchronousExecutor implements Executor {
        @Override
        public void execute(Runnable command) {
            command.run();
        }
    }

    @Test
    void testHandleMessageIsCalled() {
        MessageHandler messageHandler = Mockito.mock(MessageHandler.class);
        Executor executor = new SynchronousExecutor();
        MyClass myClass = new MyClass(executor, messageHandler);
        Message message = new Message();

        myClass.doSomething(message);

        verify(messageHandler).handleMessage(message);
    }
}

注意事项

  • 确保被测试类可以通过依赖注入的方式接收 Executor 实例。
  • SynchronousExecutor 仅适用于单元测试环境,不应在生产环境中使用。
  • 对于更复杂的异步场景,可能需要使用更高级的测试技术,例如使用 CountDownLatch 或其他同步机制来等待异步任务完成。

总结

通过使用 SynchronousExecutor,我们可以轻松地验证在 Executor.execute() 方法内部调用的方法。 这种方法简化了单元测试,并确保我们可以准确地验证异步代码的行为。 记住,这种方法主要用于单元测试,不应在生产代码中使用。在生产环境中,应该使用真实的异步 Executor 实现,并使用适当的监控和日志记录来跟踪异步任务的执行情况。

相关专题

更多
线程和进程的区别
线程和进程的区别

线程和进程的区别:线程是进程的一部分,用于实现并发和并行操作,而线程共享进程的资源,通信更方便快捷,切换开销较小。本专题为大家提供线程和进程区别相关的各种文章、以及下载和课程。

482

2023.08.10

Golang 性能分析与pprof调优实战
Golang 性能分析与pprof调优实战

本专题系统讲解 Golang 应用的性能分析与调优方法,重点覆盖 pprof 的使用方式,包括 CPU、内存、阻塞与 goroutine 分析,火焰图解读,常见性能瓶颈定位思路,以及在真实项目中进行针对性优化的实践技巧。通过案例讲解,帮助开发者掌握 用数据驱动的方式持续提升 Go 程序性能与稳定性。

9

2026.01.22

html编辑相关教程合集
html编辑相关教程合集

本专题整合了html编辑相关教程合集,阅读专题下面的文章了解更多详细内容。

56

2026.01.21

三角洲入口地址合集
三角洲入口地址合集

本专题整合了三角洲入口地址合集,阅读专题下面的文章了解更多详细内容。

28

2026.01.21

AO3中文版入口地址大全
AO3中文版入口地址大全

本专题整合了AO3中文版入口地址大全,阅读专题下面的的文章了解更多详细内容。

378

2026.01.21

妖精漫画入口地址合集
妖精漫画入口地址合集

本专题整合了妖精漫画入口地址合集,阅读专题下面的文章了解更多详细内容。

115

2026.01.21

java版本选择建议
java版本选择建议

本专题整合了java版本相关合集,阅读专题下面的文章了解更多详细内容。

3

2026.01.21

Java编译相关教程合集
Java编译相关教程合集

本专题整合了Java编译相关教程,阅读专题下面的文章了解更多详细内容。

16

2026.01.21

C++多线程相关合集
C++多线程相关合集

本专题整合了C++多线程相关教程,阅读专题下面的的文章了解更多详细内容。

9

2026.01.21

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
Kotlin 教程
Kotlin 教程

共23课时 | 2.8万人学习

C# 教程
C# 教程

共94课时 | 7.3万人学习

Java 教程
Java 教程

共578课时 | 49.4万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号