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的实例:

import java.util.function.Supplier;

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

  // 构造函数,允许外部注入B的创建逻辑
  public SomeClass(final Supplier 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();
  }
}

在这个重构后的版本中:

HeyGen
HeyGen

HeyGen是一个AI虚拟数字人生成平台,可以根据用户提供的内容,快速生成高质量的虚拟发言人视频,支持数字化身、文本转视频和视频翻译。

下载
  • 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),我们可以将依赖的创建控制权转移到外部,从而在单元测试中注入模拟对象。这种方法不仅解决了模拟难题,也促进了更健壮、更灵活的代码设计。虽然这种方法能够解决问题,但务必注意避免过度使用“模拟返回模拟”的模式,并始终将测试的重点放在验证被测单元的外部行为上,而不是其内部实现细节。良好的代码设计是实现高效、可维护单元测试的基础。

相关专题

更多
java
java

Java是一个通用术语,用于表示Java软件及其组件,包括“Java运行时环境 (JRE)”、“Java虚拟机 (JVM)”以及“插件”。php中文网还为大家带了Java相关下载资源、相关课程以及相关文章等内容,供大家免费下载使用。

832

2023.06.15

java正则表达式语法
java正则表达式语法

java正则表达式语法是一种模式匹配工具,它非常有用,可以在处理文本和字符串时快速地查找、替换、验证和提取特定的模式和数据。本专题提供java正则表达式语法的相关文章、下载和专题,供大家免费下载体验。

738

2023.07.05

java自学难吗
java自学难吗

Java自学并不难。Java语言相对于其他一些编程语言而言,有着较为简洁和易读的语法,本专题为大家提供java自学难吗相关的文章,大家可以免费体验。

734

2023.07.31

java配置jdk环境变量
java配置jdk环境变量

Java是一种广泛使用的高级编程语言,用于开发各种类型的应用程序。为了能够在计算机上正确运行和编译Java代码,需要正确配置Java Development Kit(JDK)环境变量。php中文网给大家带来了相关的教程以及文章,欢迎大家前来阅读学习。

397

2023.08.01

java保留两位小数
java保留两位小数

Java是一种广泛应用于编程领域的高级编程语言。在Java中,保留两位小数是指在进行数值计算或输出时,限制小数部分只有两位有效数字,并将多余的位数进行四舍五入或截取。php中文网给大家带来了相关的教程以及文章,欢迎大家前来阅读学习。

398

2023.08.02

java基本数据类型
java基本数据类型

java基本数据类型有:1、byte;2、short;3、int;4、long;5、float;6、double;7、char;8、boolean。本专题为大家提供java基本数据类型的相关的文章、下载、课程内容,供大家免费下载体验。

446

2023.08.02

java有什么用
java有什么用

java可以开发应用程序、移动应用、Web应用、企业级应用、嵌入式系统等方面。本专题为大家提供java有什么用的相关的文章、下载、课程内容,供大家免费下载体验。

430

2023.08.02

java在线网站
java在线网站

Java在线网站是指提供Java编程学习、实践和交流平台的网络服务。近年来,随着Java语言在软件开发领域的广泛应用,越来越多的人对Java编程感兴趣,并希望能够通过在线网站来学习和提高自己的Java编程技能。php中文网给大家带来了相关的视频、教程以及文章,欢迎大家前来学习阅读和下载。

16926

2023.08.03

C++ 单元测试与代码质量保障
C++ 单元测试与代码质量保障

本专题系统讲解 C++ 在单元测试与代码质量保障方面的实战方法,包括测试驱动开发理念、Google Test/Google Mock 的使用、测试用例设计、边界条件验证、持续集成中的自动化测试流程,以及常见代码质量问题的发现与修复。通过工程化示例,帮助开发者建立 可测试、可维护、高质量的 C++ 项目体系。

3

2026.01.16

热门下载

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

精品课程

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

共23课时 | 2.6万人学习

C# 教程
C# 教程

共94课时 | 6.8万人学习

Java 教程
Java 教程

共578课时 | 46.5万人学习

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

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