- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
当使用 async/await 函数时,调用堆栈的行为如何?
function resolveAfter2Seconds() { // taken from https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Statements/async_function
return new Promise(resolve => {
setTimeout(() => {
resolve('resolved');
}, 2000);
});
}
const asyncFuntion=async()=>{
const result = await resolveAfter2Seconds();
console.info("asyncFuntion finish, result is: ", result);
}
const first = async()=>{
await asyncFuntion();
console.log('first completed');
debugger;
}
const second = ()=>{
console.log('second completed');
debugger;
}
function main(){
first();
second();
}
main();
在上面的代码中,当在 second() 中遇到第一个断点时,我可以看到调用堆栈包含 main() 和 second()。在 first() 的第二个断点期间,调用堆栈包含 main() 和 first()。
在第一个断点期间 first() 发生了什么。它被推到哪里?假设 asyncFunction() 需要一些时间才能完成。
有人请帮助。
最佳答案
首先,当您到达您在 second
中命中的断点时,first
已经执行并且不再位于堆栈中。
当我们进入 first
时,我们立即点击了 await asyncFunction()
。这告诉 JavaScript 不要阻塞调用的结果,而是在我们等待的时候随意去寻找其他事情做。 Javascript 是做什么的?
嗯,首先,它确实调用了 asyncFunction()
。这将返回一个 promise ,并将启动一些稍后将解决它的异步过程。现在我们不能继续 first
的下一行(即 console.log('first completed')
)因为我们的 await
意味着我们不能在 promise 完成之前继续执行,所以我们需要在这里暂停执行,并找到其他空闲时间做的事情。
所以,我们查找堆栈。我们仍然在来自 main 的 first()
调用中,现在我们可以从该调用返回一个 promise ,我们将在异步执行完成时解决这个问题。 main
忽略 promise 返回值,所以我们继续 second()
。一旦我们执行了 second
,我们就会回顾调用它的任何内容,并以每个人都期望的方式继续同步执行。
然后,在未来的某个时刻,我们的 promise 就会兑现。也许它正在等待 API 调用返回。也许它正在等待数据库回复它。在我们的示例中,它正在等待 2 秒超时。无论它在等待什么,现在都可以处理了,我们可以取消挂起 first
调用并继续在那里执行。它不是从 main
中“调用”的——在内部函数被重新拾取,就像回调一样,但关键是用一个全新的堆栈调用,一旦我们完成调用其余部分,它就会被销毁的功能。
鉴于我们在一个新的堆栈中,并且早已离开“main”堆栈框架,main
和 first
如何再次出现在堆栈中当我们在里面遇到断点时?
长期以来,如果您在调试器中运行您的代码,简单的答案是它们不会。您只需获取您所在的函数,调试器就会告诉您它是从“异步代码”或类似的东西调用的。
然而,现在一些调试器可以跟踪等待的代码回到它正在解析的 promise (记住,await
和 async
主要是只是 promise 之上的语法糖)。换句话说,当您等待的代码完成,并且引擎盖下的“ promise ”得到解决时,您的调试器会帮助您找出堆栈“应该”是什么样子。它显示的内容实际上与引擎最终调用该函数的方式并没有太多相似之处——毕竟,它是在事件循环之外调用的。但是,我认为这是一个有用的补充,使我们所有人都能保持代码的思维模型比实际发生的事情简单得多!
一些关于它如何工作的进一步阅读,其中涵盖了比我在这里所能看到的更多的细节:
关于javascript - 使用 async/await 时调用堆栈的工作,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/56241258/
我有带皮肤的 DNN。我的 head 标签有 runat="server"所以我尝试在 head 标签内添加一个标签 "> 在后面的代码中,我在属性中设置了 var GoogleAPIkey。问题是它
我在 Node.JS 中有一个导出模块 exports.doSomethingImportant= function(req, res) { var id = req.params.id; Demo.
我是 F# 的新手,我一直在阅读 F# for Fun and Profit。在为什么使用 F#? 系列中,有一个 post描述异步代码。我遇到了 Async.StartChild函数,我不明白为什么
File 中有一堆相当方便的方法类,如 ReadAll***/WriteAll***/AppendAll***。 我遇到过很多情况,当我需要它们的异步对应物时,但它们根本不存在。 为什么?有什么陷阱吗
我最近开始做一个 Node 项目,并且一直在使用 async 库。我有点困惑哪个选项会更快。在某些数据上使用 async.map 并获取其结果,或使用 async.each 迭代一组用户并将他们的相应
您好,我正在试用 Springs 异步执行器,发现您可以使用 @Async。我想知道是否有可能在 @Async 中使用 @Async,要求是需要将任务委托(delegate)给 @Async 方法在第
我需要支持取消一个函数,该函数返回一个可以在启动后取消的对象。在我的例子中,requester 类位于我无法修改的第 3 方库中。 actor MyActor { ... func d
假设 asyncSendMsg不返回任何内容,我想在另一个异步块中启动它,但不等待它完成,这之间有什么区别: async { //(...async stuff...) for msg
我想用 Mocha 测试异步代码. 我跟着这个教程testing-promises-with-mocha .最后,它说最好的方法是 async/await。 以下是我的代码,我打算将 setTimeo
正如我有限(甚至错误)的理解,Async.StartImmediate 和 Async.RunSynchronously 在当前线程上启动异步计算。那么这两个功能究竟有什么区别呢?谁能帮忙解释一下?
我有一行使用await fetch() 的代码。我正在使用一些调用 eval("await fetch ...etc...") 的脚本注入(inject),但问题是 await 在执行时不会执行从ev
我正在尝试使用 nodeJS 构建一个网络抓取工具,它在网站的 HTML 中搜索图像,缓存图像源 URL,然后搜索最大尺寸的图像。 我遇到的问题是 deliverLargestImage() 在循环遍
我想结合使用 async.each 和 async.series,但得到了意想不到的结果。 async.each([1, 2], function(item, nloop) { async.s
我的代码有问题吗?我使用 async.eachSeries 但我的结果总是抛出 undefined。 这里是我的代码: async.eachSeries([1,2,3], function(data,
我想在 trait 中编写异步函数,但是因为 async fn in traits 还不被支持,我试图找到等效的方法接口(interface)。这是我在 Rust nightly (2019-01-0
async setMyPhotos() { const newPhotos = await Promise.all(newPhotoPromises); someOtherPromise();
async.js 中 async.each 与 async.every 的区别?似乎两者都相同,只是 async.every 返回结果。纠正我,我错了。 最佳答案 每个异步 .each(coll, i
我正在尝试对一组项目运行 async.each。 对于每个项目,我想运行一个 async.waterfall。请参阅下面的代码。 var ids = [1, 2]; async.each(ids,
我的目标是测试 API 调用,将延迟考虑在内。我的灵感来自 this post . 我设计了一个沙箱,其中模拟 API 需要 1000 毫秒来响应和更改全局变量 result 的值。测试检查 500
async.each 是否作为异步数组迭代工作? async.eachSeries 是否作为同步数组迭代工作?(它实际上等待响应) 我问这些是因为两者都有回调,但 async.each 的工作方式类似
我是一名优秀的程序员,十分优秀!