
在 jest 中断言模拟模块方法的调用时,常见的挑战是无法直接访问 `jest.mock()` 工厂函数内部定义的模拟函数。本文将详细介绍如何通过正确导入模块并结合 jest 的模拟机制,在 javascript 和 typescript 环境下,有效地获取并断言模拟模块方法的调用,解决“不允许引用作用域外变量”的错误。
在使用 Jest 进行模块模拟时,我们经常会遇到需要模拟一个模块的特定方法,并在测试中验证该方法是否被调用以及调用参数。一个常见的错误尝试如下:
// 错误的尝试:尝试在 jest.mock 外部定义并引用
const log = jest.fn(); // 声明在外部
jest.mock('../../../../services/logs.service.js', () => ({
    log // 在 jest.mock 工厂函数中引用外部变量
}));
// 此时,直接使用 expect(log).toHaveBeenCalledWith(...) 会导致错误
// 因为 jest.mock 的工厂函数不允许引用其作用域之外的变量。上述代码会导致 Jest 抛出错误:“The module factory of jest.mock() is not allowed to reference any out-of-scope variables.” 这是因为 Jest 的模拟机制在内部实现上,要求 jest.mock() 的第二个参数(模块工厂函数)是自包含的,不能依赖外部作用域的变量来定义模拟。
要正确地断言一个被模拟的模块方法,关键在于在测试文件中先导入原始模块的引用,然后 Jest 会在内部将这个引用指向模拟后的函数。
在 JavaScript 中,解决方案非常直接:首先从目标模块中导入你需要模拟并断言的方法,然后在 jest.mock() 中定义该方法的模拟实现。Jest 会确保你导入的变量最终指向模拟后的函数。
// 1. 从目标模块导入 'log' 方法
import { log } from '../../../../services/logs.service.js';
// 2. 使用 jest.mock 模拟整个模块,并定义 'log' 的模拟实现
// 此时,导入的 'log' 变量将指向这个 jest.fn()
jest.mock('../../../../services/logs.service.js', () => ({
    log: jest.fn() // 定义模拟函数
}));
// 3. 现在可以安全地对导入的 'log' 进行断言
describe('Log Service', () => {
    test('log method should be called with correct arguments', () => {
        // 假设这里执行了某个操作,该操作会调用 logs.service.js 中的 log 方法
        // 例如:someFunctionThatCallsLogService();
        // 模拟调用 log 方法两次
        log(2, "foo");
        log(1, "bar");
        // 断言 log 方法被调用
        expect(log).toHaveBeenCalled();
        // 断言 log 方法被调用了两次
        expect(log).toHaveBeenCalledTimes(2);
        // 断言 log 方法被调用时带有特定参数
        expect(log).toHaveBeenCalledWith(2, "foo");
        expect(log).toHaveBeenCalledWith(1, "bar");
    });
});解释: 当你在测试文件的顶部使用 import { log } from '...' 时,你获取了一个对 logs.service.js 模块中 log 导出成员的引用。当 jest.mock() 执行时(Jest 会将 jest.mock 调用提升到文件顶部),它会替换 logs.service.js 模块的 log 导出为 jest.fn()。由于你的 import 语句已经引用了那个导出成员,因此你现在通过 log 变量访问到的就是那个被模拟的 jest.fn() 实例。
在 TypeScript 中,除了上述 JavaScript 的步骤外,我们还需要进行一步类型断言,以确保 TypeScript 编译器正确识别 log 变量现在是一个 Jest 模拟函数,从而能够访问 jest.fn() 提供的匹配器(如 toHaveBeenCalledWith)。
// 1. 从目标模块导入 'log' 方法
import { log } from '../../../../services/logs.service.js';
// 2. 使用 jest.mock 模拟整个模块,并定义 'log' 的模拟实现
jest.mock('../../../../services/logs.service.js', () => ({
    log: jest.fn() // 定义模拟函数
}));
describe('Log Service (TypeScript)', () => {
    test('log method should be called with correct arguments in TS', () => {
        // 3. 对导入的 'log' 进行类型断言,明确它是一个 Jest 模拟函数
        const mockedLog = log as jest.MockedFunction<typeof log>;
        // 模拟调用 log 方法
        mockedLog(2, "foo");
        // 4. 现在可以安全地对 mockedLog 进行断言
        expect(mockedLog).toHaveBeenCalledWith(2, "foo");
        expect(mockedLog).toHaveBeenCalledTimes(1);
    });
});解释:jest.MockedFunction<typeof log> 是 Jest 提供的一个泛型类型,用于将一个函数类型转换为其模拟版本。这允许 TypeScript 编译器知道 mockedLog 变量具有 jest.fn() 实例的所有方法和属性,从而避免类型错误。
在 Jest 中断言模拟模块方法的调用,关键在于理解 jest.mock() 的作用域限制以及 Jest 如何处理模块导入。通过在测试文件中先行导入目标模块的成员,然后利用 jest.mock() 来替换这些成员的实现,我们便能获得一个指向模拟函数的引用,进而对其进行精确的调用断言。对于 TypeScript 用户,额外的类型断言步骤能够确保类型安全,并提供更好的开发体验。遵循这些实践,可以使你的 Jest 测试更加健壮和可靠。
以上就是Jest 模块模拟:如何断言被调用的方法的详细内容,更多请关注php中文网其它相关文章!
 
                        
                        每个人都需要一台速度更快、更稳定的 PC。随着时间的推移,垃圾文件、旧注册表数据和不必要的后台进程会占用资源并降低性能。幸运的是,许多工具可以让 Windows 保持平稳运行。
 
                 
                                
                                 收藏
收藏
                                                                             
                                
                                 收藏
收藏
                                                                             
                                
                                 收藏
收藏
                                                                            Copyright 2014-2025 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号