
在现代javascript应用中,异步操作无处不在。当这些异步操作在执行过程中抛出异常时,我们需要一种可靠的方式来测试这些异常是否按预期被捕获和处理。jest作为流行的javascript测试框架,提供了强大的工具来处理这类场景。然而,对于异步函数抛出异常的测试,开发者有时会混淆其正确的使用方式。
在Jest中测试一个预期会拒绝(即抛出异常)的异步函数时,我们可能会遇到以下两种常见的代码模式:
模式一:直接调用异步函数并传入其返回的 Promise
// 假设 asyncFun 是一个返回 Promise 的异步函数 await expect(asyncFun()).rejects.toThrowError(errorObj);
模式二:将异步函数的调用包裹在一个新的异步函数中
// 假设 asyncFun 是一个返回 Promise 的异步函数
await expect(async () => {
await asyncFun();
}).rejects.toThrowError(errorObj);那么,这两种模式是否存在差异,哪一种是推荐的用法呢?答案是肯定的,它们之间存在显著差异,并且模式一才是Jest官方文档推荐且设计意图所在的用法。
理解这两种模式差异的关键在于深入理解Jest的 rejects 匹配器的工作原理。
Jest的 rejects 匹配器专门用于测试 Promise 是否被拒绝。它的设计初衷是接收一个 Promise 对象作为 expect() 的参数。当 expect() 接收到一个 Promise 时,rejects 会等待该 Promise 的状态变为拒绝(rejected),然后将拒绝的原因与后续的匹配器(如 toThrowError)进行比较。
与同步 toThrow() 的对比: 为了更好地理解 rejects,我们可以将其与同步的 toThrow() 匹配器进行对比。toThrow() 匹配器用于测试同步函数是否抛出异常。它的用法是 expect(() => { /* 同步函数调用 */ }).toThrowError(errorObj)。在这里,expect() 接收的是一个函数,Jest会执行这个函数,并捕获它可能抛出的同步异常。
模式二的问题所在: 模式二 await expect(async () => { await asyncFun() }).rejects.toThrowError(errorObj) 的问题在于,它将一个函数(async () => { await asyncFun() })传递给了 expect(),而不是一个 Promise。尽管在某些Jest版本或特定场景下,这种写法可能“看起来”能工作,但它并非 rejects 匹配器的设计意图。rejects 期望直接操作一个 Promise,而不是去执行一个函数来获取 Promise。这种不符合设计意图的用法可能导致:
基于上述分析,测试异步函数抛出异常的正确且推荐的方式是模式一:直接将异步函数调用后返回的 Promise 传递给 expect()。
示例代码:
// 假设我们有一个异步函数,它在某些条件下会抛出错误
async function fetchData(shouldFail) {
return new Promise((resolve, reject) => {
setTimeout(() => {
if (shouldFail) {
reject(new Error('数据加载失败!'));
} else {
resolve('数据加载成功!');
}
}, 100);
});
}
// 异步函数示例,使用 async/await 语法
async function processData(input) {
if (input === 'error') {
throw new Error('无效的输入');
}
return `处理结果: ${input}`;
}
describe('异步函数异常测试', () => {
test('fetchData 函数应在失败时拒绝并抛出特定错误', async () => {
// 正确用法:直接调用 fetchData(),它返回一个 Promise
await expect(fetchData(true)).rejects.toThrowError('数据加载失败!');
await expect(fetchData(true)).rejects.toThrowError(Error); // 也可以检查错误类型
});
test('processData 函数应在输入错误时抛出异常', async () => {
// 正确用法:直接调用 processData(),它返回一个 Promise
await expect(processData('error')).rejects.toThrowError('无效的输入');
await expect(processData('error')).rejects.toThrowError(new Error('无效的输入'));
});
// 错误用法示例(不推荐)
test('不推荐的异步异常测试模式', async () => {
// 尽管这个测试可能通过,但它不是 Jest rejects 匹配器的设计意图
// await expect(async () => {
// await fetchData(true);
// }).rejects.toThrowError('数据加载失败!');
//
// await expect(async () => {
// await processData('error');
// }).rejects.toThrowError('无效的输入');
});
test('fetchData 函数在成功时应解决', async () => {
await expect(fetchData(false)).resolves.toBe('数据加载成功!');
});
});在上面的示例中,await expect(fetchData(true)) 和 await expect(processData('error')) 是正确的用法。fetchData(true) 和 processData('error') 会立即被调用并返回一个 Promise,然后这个 Promise 被传递给 expect(),rejects 匹配器会等待它被拒绝。
在Jest中测试异步函数抛出的异常时,关键在于将异步函数调用后返回的 Promise 对象直接传递给 expect(),并结合 rejects.toThrowError() 匹配器。避免将异步函数调用包裹在另一个函数中传递给 expect(),这虽然可能在某些情况下“奏效”,但并非Jest rejects 匹配器的正确用法,可能导致测试行为的不一致或未来兼容性问题。遵循官方推荐的模式,能够确保测试的健壮性、可读性,并与Jest的设计哲学保持一致。
以上就是Jest中测试异步函数抛出异常:rejects 的正确用法解析的详细内容,更多请关注php中文网其它相关文章!
每个人都需要一台速度更快、更稳定的 PC。随着时间的推移,垃圾文件、旧注册表数据和不必要的后台进程会占用资源并降低性能。幸运的是,许多工具可以让 Windows 保持平稳运行。
Copyright 2014-2025 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号