在Node.js中使用async-await时,为什么为什么松开堆栈跟踪?

时间:2019-03-14 12:31:34

标签: javascript node.js error-handling async-await stack-trace

当我运行以下程序时

async function functionOne() {

  throw new Error('Error here prints the complete stack');

  await new Promise((resolve) => {
    setTimeout(() => { resolve(); }, 1000);
  });
}

async function functionTwo() {
  await functionOne();
}

async function functionThree() {
  await functionTwo();
}

functionThree()
  .catch((error) => {
    console.error(error);
  });

我得到以下输出,该输出通过各种调用的函数打印堆栈

Error: Error here prints the complete stack
    at functionOne (/home/divyanshu/programming/errorHandlingAsyncAwait/index.js:3:9)
    at functionTwo (/home/divyanshu/programming/errorHandlingAsyncAwait/index.js:11:9)
    at functionThree (/home/divyanshu/programming/errorHandlingAsyncAwait/index.js:15:9)
    at Object.<anonymous> (/home/divyanshu/programming/errorHandlingAsyncAwait/index.js:18:1)
    at Module._compile (module.js:612:30)
    at Object.Module._extensions..js (module.js:623:10)
    at Module.load (module.js:531:32)
    at tryModuleLoad (module.js:494:12)
    at Function.Module._load (module.js:486:3)
    at Function.Module.runMain (module.js:653:10)

但是在以下程序中的await调用后引发错误时

async function functionOne() {

  await new Promise((resolve) => {
    setTimeout(() => { resolve(); }, 1000);
  });

  throw new Error('Error here prints incomplete stack');

}

async function functionTwo() {
  await functionOne();
}

async function functionThree() {
  await functionTwo();
}

functionThree()
  .catch((error) => {
    console.error(error);
  });

这是输出

Error: Error here prints incomplete stack
    at functionOne (/home/divyanshu/programming/errorHandlingAsyncAwait/index.js:7:9)
    at <anonymous>

我想了解为什么在第二种情况下而不在第一种情况下丢失堆栈跟踪。

1 个答案:

答案 0 :(得分:2)

因为在第一个代码中,直到Error为止的所有内容都在事件循环的同一时刻。

在异步回调之前,setTimeout中的一个可以进入调用堆栈(堆栈跟踪是从该堆栈建立的),调用堆栈必须为空。

因此,由于第一个代码会同步运行所有内容,直到进行Error调用,因此所有这些调用都在调用堆栈中。但是在第二种方法中,Error是在await调用setTimeout之后被调用的。 setTimeout完成后。事件循环将回调放回堆栈中,为此,调用堆栈必须为空。

所以现在,调用堆栈上没有functionTwofunctionThree,这就是为什么它们不出现的原因。

T 堆栈跟踪是发生错误时堆栈的状态。

以下是两种代码中堆栈发生的情况的粗略解释:

第一个密码

1) functionThree is pushed into the stack
2) functionTwo is pushed into the stack
3) functionOne is pushed into the stack
4) Error is thrown

第二个代码

1) functionThree is pushed into the stack
2) functionTwo is pushed into the stack
3) functionOne is pushed into the stack
4) awat ...setTimeout is called
5) All 3 functions return a Promise
6) The stack is empty

... setTimeout ends
Next tick of the event loop

1) setTimeout callback is called
2) Error is thrown

我建议观看此视频以了解所有操作原理:

Philip Roberts的

What the heck is the event loop anyway


在最新版本的Node.js上,使用异步代码时,可以使用--async-stack-traces标志来改善堆栈跟踪。

您可以在https://v8.dev/blog/fast-async

上了解更多信息。