0

0

Mockito实践:如何优雅地模拟内部创建对象及其方法返回结果

花韻仙語

花韻仙語

发布时间:2025-11-13 22:05:15

|

203人浏览过

|

来源于php中文网

原创

Mockito实践:如何优雅地模拟内部创建对象及其方法返回结果

本文探讨了在使用mockito进行单元测试时,如何模拟由内部创建对象的方法返回的对象。当被测类与依赖对象紧密耦合时,直接模拟会失败。文章通过重构代码,引入依赖注入或工厂模式,使得内部依赖可被测试框架控制,从而实现对返回对象的有效模拟,并强调了测试中避免过度使用模拟对象的重要性。

1. 理解内部依赖模拟的挑战

在Java中使用Mockito进行单元测试时,一个常见的挑战是模拟那些在被测类内部直接实例化(通过new关键字)的依赖对象,以及这些对象的方法所返回的结果。考虑以下代码结构:

class SomeClass {
    public void doSomeThing() {
        B b = new B(); // 内部创建B的实例
        A a = b.foo(); // 调用B的方法,返回A的实例
        a.foo();       // 对返回的A实例进行操作
    }
}

在上述SomeClass中,B的实例是在doSomeThing方法内部创建的。这意味着SomeClass与B的具体实现紧密耦合。如果我们尝试使用传统的Mockito方式来模拟A,例如:

@Mock
A a; // 这是一个A的模拟对象

@InjectMock
SomeClass someClass;

@Test
void test() {
    // 尝试配置a的foo()方法行为
    Mockito.when( a.foo() ).thenReturn( something ); 

    assertDoesNotThrow( () -> someClass.doSomeThing() );
}

这种测试方法将无法奏效。原因在于,@Mock A a创建的模拟对象与SomeClass内部通过new B().foo()实际获取到的A对象是完全不同的两个实例。SomeClass内部仍然会调用真实的B实例的foo()方法,并获得一个真实的A实例(或者一个未被模拟的A实例),而不是我们期望的模拟A对象。因此,对@Mock A a的配置对SomeClass内部的执行没有任何影响。被测类与依赖的紧密耦合阻止了测试框架介入依赖对象的创建过程。

2. 重构以提升可测试性:引入依赖注入

要解决上述问题,核心在于打破SomeClass与B之间的紧密耦合,使得B的创建过程可以被外部控制,从而在测试时能够注入一个模拟的B实例。一种有效的策略是使用依赖注入(Dependency Injection)模式,或者更具体地,通过提供一个工厂或Supplier来控制依赖的创建。

我们可以将SomeClass重构如下,允许外部注入一个Supplier来提供B的实例:

SumiNote
SumiNote

一款服务留学生的AI学习神器

下载
import java.util.function.Supplier;

class SomeClass {
  private final Supplier<? extends B> bFactory; // 使用Supplier来提供B的实例

  // 构造函数,允许外部注入B的创建逻辑
  public SomeClass(final Supplier<? extends B> bFactory) {
    this.bFactory = bFactory;
  }

  // 无参构造函数,为现有代码提供兼容性,实际应用中建议统一使用带参构造函数
  public SomeClass() {
    this(B::new); // 默认行为:创建真实的B实例
  }

  public void doSomeThing() {
    B b = this.bFactory.get(); // 从Supplier获取B的实例
    A a = b.foo();
    a.foo();
  }
}

在这个重构后的版本中:

  • SomeClass不再直接使用new B()来创建B的实例,而是通过一个Supplier接口来获取。
  • 在生产代码中,可以通过new SomeClass(B::new)来保持原有的行为。
  • 更推荐的做法是,所有依赖SomeClass的地方都通过构造函数注入其所需的Supplier,这进一步提升了模块间的解耦。这种设计遵循了依赖倒置原则,使得SomeClass不再依赖于B的具体实现,而是依赖于一个抽象的Supplier接口。

3. 实施模拟策略

有了重构后的SomeClass,我们现在可以轻松地在测试中注入一个模拟的B实例,进而控制B.foo()的返回值。

import org.junit.jupiter.api.Test;
import static org.mockito.Mockito.*;
import static org.junit.jupiter.api.Assertions.assertDoesNotThrow;

// 假设A和B是已定义的类,此处为示例实现
class A { 
    public void foo() { System.out.println("A's real foo called"); } 
}
class B { 
    public A foo() { 
        System.out.println("B's real foo called");
        return new A(); 
    } 
} 

class SomeClassTest {

    @Test
    void testDoSomeThingWithMockedDependencies() {
        // 1. 创建A的模拟对象
        final A aMock = mock(A.class);
        // 配置aMock.foo()的行为(如果aMock.foo()会被调用)。
        // 例如,让它什么都不做,或者抛出异常等。
        doNothing().when(aMock).foo(); 

        // 2. 创建B的模拟对象
        final B bMock = mock(B.class);
        // 配置bMock.foo()的行为,使其返回aMock
        when(bMock.foo()).thenReturn(aMock);

        // 3. 创建SomeClass的实例,并注入一个Supplier,该Supplier返回bMock
        // 这样,SomeClass在执行doSomeThing时,会通过bFactory.get()获取到bMock
        final SomeClass someClass = new SomeClass(() -> bMock);

        // 4. 执行被测方法
        assertDoesNotThrow(() -> someClass.doSomeThing());

        // 5. 验证交互(可选):确保方法按预期被调用
        verify(bMock).foo(); // 验证bMock的foo方法是否被调用
        verify(aMock).foo(); // 验证aMock的foo方法是否被调用
    }
}

在这个测试中,我们:

  1. 创建了一个A的模拟对象aMock,并配置了其foo()方法的行为。
  2. 创建了一个B的模拟对象bMock。
  3. 配置bMock.foo()方法在被调用时返回aMock。
  4. 实例化SomeClass时,传入一个Lambda表达式() -> bMock作为Supplier,这样SomeClass内部在需要B实例时,就会得到我们提供的bMock。 通过这种方式,我们成功地控制了SomeClass内部对B和A的依赖,实现了对内部创建对象及其返回结果的模拟。

4. 注意事项与最佳实践

尽管上述方法能够解决内部依赖的模拟问题,但在实际测试中,有几个重要的注意事项和最佳实践需要牢记:

  1. 避免“模拟返回模拟” (Mocks returning Mocks): 在上述示例中,我们让bMock返回了aMock。虽然这在某些情况下是必要的,但通常被认为是一种“代码异味”(code smell)。过多的“模拟返回模拟”会使测试变得:

    • 脆弱 (Brittle):测试与实现细节过度耦合。如果B.foo()的返回类型或行为在未来发生变化,即使SomeClass的核心业务逻辑没有改变,测试也可能失败。
    • 复杂 (Complex):增加了测试代码的阅读和维护难度。
    • 难以理解 (Hard to understand):模糊了测试的意图,使得测试不再清晰地表达被测单元的行为。 在设计时,如果发现需要频繁地进行“模拟返回模拟”,这可能暗示着被测单元(如SomeClass)的职责过于庞大,或者其依赖关系过于复杂。此时,应考虑对代码进行进一步的解耦或重构。例如,SomeClass是否真的需要直接操作A的实例?它是否可以将对A的操作委托给另一个服务?
  2. 专注于被测单元的行为: 单元测试的目标是验证单个单元(通常是一个类或一个方法)的特定行为,而不是其内部的实现细节。当模拟层级过深时,我们实际上可能在测试多个单元的协作,这更像是集成测试的范畴。尽量使模拟对象只模拟其直接依赖的行为。

  3. 重构的价值: 本教程的核心在于通过重构(引入Supplier或依赖注入)来提升代码的可测试性。这种重构不仅有助于单元测试,还能提高代码的模块化程度、可维护性和灵活性。一个设计良好的类应该易于测试,而易于测试的类往往也是设计良好的。

总结

要有效地模拟由内部创建对象的方法返回的对象,关键在于打破被测类与这些内部依赖之间的紧密耦合。通过引入依赖注入或工厂模式(如Supplier),我们可以将依赖的创建控制权转移到外部,从而在单元测试中注入模拟对象。这种方法不仅解决了模拟难题,也促进了更健壮、更灵活的代码设计。虽然这种方法能够解决问题,但务必注意避免过度使用“模拟返回模拟”的模式,并始终将测试的重点放在验证被测单元的外部行为上,而不是其内部实现细节。良好的代码设计是实现高效、可维护单元测试的基础。

热门AI工具

更多
DeepSeek
DeepSeek

幻方量化公司旗下的开源大模型平台

豆包大模型
豆包大模型

字节跳动自主研发的一系列大型语言模型

通义千问
通义千问

阿里巴巴推出的全能AI助手

腾讯元宝
腾讯元宝

腾讯混元平台推出的AI助手

文心一言
文心一言

文心一言是百度开发的AI聊天机器人,通过对话可以生成各种形式的内容。

讯飞写作
讯飞写作

基于讯飞星火大模型的AI写作工具,可以快速生成新闻稿件、品宣文案、工作总结、心得体会等各种文文稿

即梦AI
即梦AI

一站式AI创作平台,免费AI图片和视频生成。

ChatGPT
ChatGPT

最最强大的AI聊天机器人程序,ChatGPT不单是聊天机器人,还能进行撰写邮件、视频脚本、文案、翻译、代码等任务。

相关专题

更多
lambda表达式
lambda表达式

Lambda表达式是一种匿名函数的简洁表示方式,它可以在需要函数作为参数的地方使用,并提供了一种更简洁、更灵活的编码方式,其语法为“lambda 参数列表: 表达式”,参数列表是函数的参数,可以包含一个或多个参数,用逗号分隔,表达式是函数的执行体,用于定义函数的具体操作。本专题为大家提供lambda表达式相关的文章、下载、课程内容,供大家免费下载体验。

214

2023.09.15

python lambda函数
python lambda函数

本专题整合了python lambda函数用法详解,阅读专题下面的文章了解更多详细内容。

192

2025.11.08

Python lambda详解
Python lambda详解

本专题整合了Python lambda函数相关教程,阅读下面的文章了解更多详细内容。

60

2026.01.05

硬盘接口类型介绍
硬盘接口类型介绍

硬盘接口类型有IDE、SATA、SCSI、Fibre Channel、USB、eSATA、mSATA、PCIe等等。详细介绍:1、IDE接口是一种并行接口,主要用于连接硬盘和光驱等设备,它主要有两种类型:ATA和ATAPI,IDE接口已经逐渐被SATA接口;2、SATA接口是一种串行接口,相较于IDE接口,它具有更高的传输速度、更低的功耗和更小的体积;3、SCSI接口等等。

1748

2023.10.19

PHP接口编写教程
PHP接口编写教程

本专题整合了PHP接口编写教程,阅读专题下面的文章了解更多详细内容。

569

2025.10.17

php8.4实现接口限流的教程
php8.4实现接口限流的教程

PHP8.4本身不内置限流功能,需借助Redis(令牌桶)或Swoole(漏桶)实现;文件锁因I/O瓶颈、无跨机共享、秒级精度等缺陷不适用高并发场景。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

2338

2025.12.29

java接口相关教程
java接口相关教程

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

43

2026.01.19

Golang 测试体系与代码质量保障:工程级可靠性建设
Golang 测试体系与代码质量保障:工程级可靠性建设

Go语言测试体系与代码质量保障聚焦于构建工程级可靠性系统。本专题深入解析Go的测试工具链(如go test)、单元测试、集成测试及端到端测试实践,结合代码覆盖率分析、静态代码扫描(如go vet)和动态分析工具,建立全链路质量监控机制。通过自动化测试框架、持续集成(CI)流水线配置及代码审查规范,实现测试用例管理、缺陷追踪与质量门禁控制,确保代码健壮性与可维护性,为高可靠性工程系统提供质量保障。

43

2026.02.28

Golang 工程化架构设计:可维护与可演进系统构建
Golang 工程化架构设计:可维护与可演进系统构建

Go语言工程化架构设计专注于构建高可维护性、可演进的企业级系统。本专题深入探讨Go项目的目录结构设计、模块划分、依赖管理等核心架构原则,涵盖微服务架构、领域驱动设计(DDD)在Go中的实践应用。通过实战案例解析接口抽象、错误处理、配置管理、日志监控等关键工程化技术,帮助开发者掌握构建稳定、可扩展Go应用的最佳实践方法。

38

2026.02.28

热门下载

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

精品课程

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

共23课时 | 4.1万人学习

C# 教程
C# 教程

共94课时 | 10.5万人学习

Java 教程
Java 教程

共578课时 | 75.7万人学习

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

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