你是否运行过不按你预期运行的 js 代码 ?
比如:某个函数被随机的、不可预测时间的执行了,或者被延迟执行了。
这时,你需要从 ES6 中引入的一个非常酷的新特性: Promise 来处理你的问题。
为了深入理解 Promise ,我在某个不眠之夜,做了一些动画来演示 Promise 的运行,我多年来的好奇心终于得到实现。
对于 Promise ,您为什么要使用它,它在底层是如何工作的,以及我们如何以最现代的方式编写它呢?
在书写 JavaScript 的时候,我们经常不得不去处理一些依赖于其它任务的任务!
比如:我们想要得到一个图片,对其进行压缩,应用一个滤镜,然后保存它 。
首先,先用 getImage 函数要得到我们想要编辑的图片。
一旦图片被成功加载,把这个图片值传到一个 ocmpressImage 函数中。
当图片已经被成功地重新调整大小后,在 applyFilter 函数中为图片应用一个滤镜。
在图片被压缩和添加滤镜后,保存图片并且打印成功的日志!
最后,代码很简单如图:
注意到了吗?尽管以上代码也能得到我们想要的结果,但是完成的过程并不是友好。
使用了大量嵌套的回调函数,这使我们的代码阅读起来特别困难。
因为写了许多嵌套的回调函数,这些回调函数又依赖于前一个回调函数,这通常被称为 回调地狱。
幸运的,ES6 中的 Promise 的能很好的处理这种情况!
让我们看看 promise 是什么,以及它是如何在类似于上述的情况下帮助我们的。
ES6引入了Promise。在许多教程中,你可能会读到这样的内容:
Promise 是一个值的占位符,这个值在未来的某个时间要么 resolve 要么 reject 。
对于我来说,这样的解释从没有让事情变得更清楚。
事实上,它只是让我感觉 Promise 是一个奇怪的、模糊的、不可预测的一段魔法。
接下来让我们看看 promise 真正是什么?
我们可以使用一个接收一个回调函数的 Promise 构造器创建一个 promise。
好酷,让我们尝试一下!
等等,刚刚得到的返回值是什么?
Promise 是一个对象,它包含一个状态 PromiseStatus
和一个值 PromiseValue
。
在上面的例子中,你可以看到 PromiseStatus
的值是 pending, PromiseValue
的值是 undefined。
不过 - 你将永远不会与这个对象进行交互,你甚至不能访问 PromiseStatus
和 PromiseValue
这两个属性!
然而,在使用 Promise 的时候,这俩个属性的值是非常重要的。
PromiseStatus
的值,也就是 Promise
的状态,可以是以下三个值之一:
fulfilled
: promise
已经被 resolved
。一切都很好,在 promise
内部没有错误发生。rejected
: promise
已经被 rejected
。哎呦,某些事情出错了。pending
: promise
暂时还没有被解决也没有被拒绝,仍然处于 pending
状态好吧,这一切听起来很棒,但是什么时候 promise
的状态是 pending
、fulfilled
或 rejected
呢? 为什么这个状态很重要呢?
在上面的例子中,我们只是为 Promise
构造器传递了一个简单的回调函数 () => {}
。
然而,这个回调函数实际上接受两个参数。
resolve
或 res
,它是一个函数,在 Promise
应该解决 resolve
的时候会被调用。reject
或rej
,它也是一个函数,在 Promise
出现一些错误应该被拒绝 reject
的时候被调用。
让我们尝试看看当我们调用 resolve
或 reject
方法时得到的日志。
在我的例子中,把 resolve
方法叫做 res
,把 reject
方法叫做 rej
。
太好了!我们终于知道如何摆脱 pending
状态和 undefined
值了!
resolve
方法时,promise
的状态是 fulfilled
。reject
方法时,promise
的状态是 rejected
。有趣的是,我让(Jake Archibald)校对了这篇文章,他实际上指出 Chrome 中存在一个错误,该错误当前将状态显示为 “ fulfilled” 而不是 “ resolved”。感谢 Mathias Bynens,它现已在Canary 中修复!
好了,现在我们知道如何更好控制那个模糊的 Promise 对象。但是他被用来做什么呢?
在前面的介绍章节,我展示了一个获得图片、压缩图片、为图片应用过滤器并保存它的例子!最终,这变成了一个混乱的嵌套回调。
幸运的,Promise
可以帮助我们解决这个问题!
首先,让我们重写整个代码块,以便每个函数返回一个 Promise
来代替之前的函数。
如果图片被加载完成并且一切正常,让我们用加载完的图片解决 (resolve)promise
。
否则,如果在加载文件时某个地方有一个错误,我们将会用发生的错误拒绝 (reject)promise
。
让我们看下当我们在终端运行这段代码时会发生什么?
非常酷!就像我们所期望的一样,promise
得到了解析数据后的值。
但是现在呢?我们不关心整个 promise
对象,我们只关心数据的值!幸运的,有内置的方法来得到 promise
的值。
对于一个 promise
,我们可以使用它上面的 3 个方法:
.then 方法接收传递给 resolve
方法的值。
.catch 方法接收传递给 rejected
方法的值。
最终,我们拥有了 promise
被解决后 (resolved)
的值,并不需要整个 promise
对象!
现在我们可以用这个值做任何我们想做的事。
顺便提醒一下,当你知道一个 promise
总是 resolve
或者总是 reject
的时候,你可以写 Promise.resolve
或 Promise.reject
,传入你想要 reject
或 resolve
的 promise
的值。
在下边的例子中你将会经常看到这个语法。
在 getImage 的例子中,为了运行它们,我们最终不得不嵌套多个回调。幸运的,.then
处理器可以帮助我们完成这件事!
.then
它自己的执行结果是一个 promise
。这意味着我们可以链接任意数量的 .then
:前一个 then
回调的结果将会作为参数传递给下一个 then
回调!
在 getImage 示例中,为了传递被处理的图片到下一个函数,我们可以链接多个 then
回调。
相比于之前最终得到许多嵌套回调,现在我们得到了整洁的 then
链。
完美!这个语法看起来已经比之前的嵌套回调好多了。
我们知道了一些如何创建 promise
以及如何提取出 promise
的值的方法。
让我们为脚本添加一些更多的代码并且再次运行它:
等下,发生了什么?!
首先,Start!
被输出。
好的,我们已经看到了那一个即将到来的消息:console.log('Start!')
在最前一行输出!
然而,第二个被打印的值是 End!
,并不是 promise
被解决的值!只有在 End!
被打印之后,promise
的值才会被打印。
这里发生了什么?
我们最终看到了 promise
真正的力量!尽管 JavaScript
是单线程的,我们可以使用 Promise
添加异步任务!
等等,我们之前没见过这种情况吗?
在 JavaScript Event Loop 中,我们不是也可以使用浏览器原生的方法如 setTimeout
创建某类异步行为吗?
是的!然而,在事件循环内部,实际上有 2 种类型的队列:宏任务(macro)队列 (或者只是叫做 任务队列 )和 微任务队列。
(宏)任务队列用于 宏任务,微任务队列用于 微任务。
那么什么是宏任务,什么是微任务呢?
尽管他们比我在这里介绍的要多一些,但是最常用的已经被展示在下面的表格中!
(Macro)task: | setTimeout | setInterval | setImmediate |
Microtask: | process.nextTick | Promise callback | queueMicrotask |
我们看到 Promise
在微任务列表中!当一个 Promise
解决 (resolve
) 并且调用它的 then()
、catch()
或 finally()
方法的时候,这些方法里的回调函数被添加到微任务队列!
这意味着 then(),chatch() 或 finally()
方法内的回调函数不是立即被执行,本质上是为我们的 JavaScript
代码添加了一些异步行为!
那么什么时候执行 then(),catch(),或 finally() 内的回调呢?
事件循环给与任务不同的优先级:
让我们快速地看一个简单的例子:
promise
中 then
方法里的回调,或者用 queueMicrotask
添加的一个任务。setTimeout 或者 setImmediate
里的回调首先,Task1 返回一个值并且从调用栈中弹出。然后,JavaScript
引擎检查微任务队列中排队的任务。一旦微任务中所有的任务被放入调用栈并且最终被弹出,JavaScript 引擎会检查宏任务队列中的任务,将他们弹入调用栈中并且在它们返回值的时候把它们弹出调用栈。
图中足够粉色的盒子是不同的任务,让我们用一些真实的代码来使用它!
在这段代码中,我们有宏任务 setTimeout 和 微任务 promise 的 then 回调。
一旦 JavaScript 引擎到达 setTimeout 函数所在的那行就会涉及到事件循环。
让我们一步一步地运行这段代码,看看会得到什么样的日志!
快速提一下:在下边的例子中,我正在展示的像
console.log
,setTimeout
和Promise.resolve
等方法正在被添加到调用栈中。它们是内部的方法实际上没有出现在堆栈痕迹中,因此如果你正在使用调试器,不用担心,你不会在任何地方见到它们。它只是在没有添加一堆样本文件代码的情况下使这个概念解释起来更加简单。
在第一行,JavaScript
引擎遇到了 console.log()
方法,它被添加到调用栈,之后它在控制台输出值 Start!。console.log
函数从调用栈内弹出,之后 JavaScript
引擎继续执行代码。
JavaScript
引擎遇到了 setTimeout
方法,他被弹入调用栈中。setTimeout
是浏览器的原生方法:它的回调函数 (() => console.log('In timeout'))
将会被添加到 Web API
,直到计时器完成计时。尽管我们为计时器提供的值是 0,在它被添加到宏任务队列 (setTimeout
是一个宏任务) 之后回调还是会被首先推入 Web API
。
JavaScript
引擎遇到了 Promise.resolve
方法。Promise.resolve
被添加到调用栈。在 Promise
解决 (resolve
) 值之后,它的 then
中的回调函数被添加到微任务队列。
JavaScript
引擎看到调用栈现在是空的。由于调用栈是空的,它将会去检查在微任务队列中是否有在排队的任务!是的,有任务在排队,promise
的 then
中的回调函数正在等待轮到它!它被弹入调用栈,之后它输出了 promise
被解决后( resolved
)的值: 在这个例子中的字符串 Promise!
。
JavaScript
引擎看到调用栈是空的,因此,如果任务在排队的话,它将会再次去检查微任务队列。此时,微任务队列完全是空的。
到了去检查宏任务队列的时候了:setTimeout
回调仍然在那里等待!setTimeout
被弹入调用栈。回调函数返回 console.log
方法,输出了字符串 In timeout!
。setTimeout
回调从调用栈中弹出。
终于,所有的事情完成了! 看起来我们之前看到的输出最终并不是那么出乎意料。
ES7
引入了一个新的在 JavaScript
中添加异步行为的方式并且使 promise
用起来更加简单!随着 async
和 await
关键字的引入,我们能够创建一个隐式的返回一个 promise
的 async
函数。但是,我们该怎么做呢?
之前,我们看到不管是通过输入 new Promise(() => {})
,Promise.resolve
或 Promise.reject
,我们都可以显式的使用 Promise
对象创建 promise
。
我们现在能够创建隐式地返回一个对象的异步函数,而不是显式地使用 Promise
对象!这意味着我们不再需要写任何 Promise
对象了。
尽管 async
函数隐式的返回 promise
是一个非常棒的事实,但是在使用 await
关键字的时候才能看到 async
函数的真正力量。当我们等待 await
后的值返回一个 resolved
的 promise
时,通过 await
关键字,我们可以暂停异步函数。如果我们想要得到这个 resolved
的 promise
的值,就像我们之前用 then
回调那样,我们可以为被 await
的 promise
的值赋值为变量!
这样,我们就可以暂停一个异步函数吗?很好,但这到底是什么意思?
当我们运行下面的代码块时让我们看下发生了什么:
额,这里发生了什么呢?
首先,JavaScript
引擎遇到了 console.log
。它被弹入到调用栈中,这之后 Before function!
被输出。
然后,我们调用了异步函数myFunc()
,这之后myFunc
函数体运行。函数主体内的最开始一行,我们调用了另一个console.log
,这次传入的是字符串In function!
。console.log
被添加到调用栈中,输出值,然后从栈内弹出。
函数体继续执行,将我们带到第二行。最终,我们看到一个await
关键字!
最先发生的事是被等待的值执行:在这个例子中是函数one
。它被弹入调用栈,并且最终返回一个解决状态的promise
。一旦Promise
被解决并且one
返回一个值,JavaScript
遇到了await
关键字。
当遇到await
关键字的时候,异步函数被暂停。函数体的执行被暂停,async
函数中剩余的代码会在微任务中运行而不是一个常规任务!
现在,因为遇到了await
关键字,异步函数myFunc
被暂停,JavaScript
引擎跳出异步函数,并且在异步函数被调用的执行上下文中继续执行代码:在这个例子中是全局执行上下文!♀️
最终,没有更多的任务在全局执行上下文中运行!事件循环检查看看是否有任何的微任务在排队:是的,有!在解决了one
的值以后,异步函数myFunc
开始排队。myFunc
被弹入调用栈中,在它之前中断的地方继续运行。
变量res
最终获得了它的值,也就是one
返回的promise
被解决的值!我们用res
的值(在这个例子中是字符串One!
)调用console.log
。One!
被打印到控制台并且console.log
从调用栈弹出。
最终,所有的事情都完成了!你注意到async
函数相比于promise
的then
有什么不同吗?await
关键字暂停了async
函数,然而如果我们使用then
的话,Promise
的主体将会继续被执行!
嗯,这是相当多的信息!当使用Promise
的时候,如果你仍然感觉有一点不知所措,完全不用担心。我个人认为,当使用异步JavaScript
的时候,只是需要经验去注意模式之后便会感到自信。
当使用异步JavaScript
的时候,我希望你可能遇到的“无法预料的”或“不可预测的”行为现在变得更有意义!
外国友人技术博客的语言表达的方式和风格、与国人的还是有很大差别的啊。
每每看到有很长或者很拗口的句子的时候,我就想按自己的语言来写一篇了
可能自己写一篇都比翻译的快
本文由哈喽比特于3年以前收录,如有侵权请联系我们。
文章来源:https://mp.weixin.qq.com/s/JE1mYeGgSXK5JZDFUh2W1A
京东创始人刘强东和其妻子章泽天最近成为了互联网舆论关注的焦点。有关他们“移民美国”和在美国购买豪宅的传言在互联网上广泛传播。然而,京东官方通过微博发言人发布的消息澄清了这些传言,称这些言论纯属虚假信息和蓄意捏造。
日前,据博主“@超能数码君老周”爆料,国内三大运营商中国移动、中国电信和中国联通预计将集体采购百万台规模的华为Mate60系列手机。
据报道,荷兰半导体设备公司ASML正看到美国对华遏制政策的负面影响。阿斯麦(ASML)CEO彼得·温宁克在一档电视节目中分享了他对中国大陆问题以及该公司面临的出口管制和保护主义的看法。彼得曾在多个场合表达了他对出口管制以及中荷经济关系的担忧。
今年早些时候,抖音悄然上线了一款名为“青桃”的 App,Slogan 为“看见你的热爱”,根据应用介绍可知,“青桃”是一个属于年轻人的兴趣知识视频平台,由抖音官方出品的中长视频关联版本,整体风格有些类似B站。
日前,威马汽车首席数据官梅松林转发了一份“世界各国地区拥车率排行榜”,同时,他发文表示:中国汽车普及率低于非洲国家尼日利亚,每百户家庭仅17户有车。意大利世界排名第一,每十户中九户有车。
近日,一项新的研究发现,维生素 C 和 E 等抗氧化剂会激活一种机制,刺激癌症肿瘤中新血管的生长,帮助它们生长和扩散。
据媒体援引消息人士报道,苹果公司正在测试使用3D打印技术来生产其智能手表的钢质底盘。消息传出后,3D系统一度大涨超10%,不过截至周三收盘,该股涨幅回落至2%以内。
9月2日,坐拥千万粉丝的网红主播“秀才”账号被封禁,在社交媒体平台上引发热议。平台相关负责人表示,“秀才”账号违反平台相关规定,已封禁。据知情人士透露,秀才近期被举报存在违法行为,这可能是他被封禁的部分原因。据悉,“秀才”年龄39岁,是安徽省亳州市蒙城县人,抖音网红,粉丝数量超1200万。他曾被称为“中老年...
9月3日消息,亚马逊的一些股东,包括持有该公司股票的一家养老基金,日前对亚马逊、其创始人贝索斯和其董事会提起诉讼,指控他们在为 Project Kuiper 卫星星座项目购买发射服务时“违反了信义义务”。
据消息,为推广自家应用,苹果现推出了一个名为“Apps by Apple”的网站,展示了苹果为旗下产品(如 iPhone、iPad、Apple Watch、Mac 和 Apple TV)开发的各种应用程序。
特斯拉本周在美国大幅下调Model S和X售价,引发了该公司一些最坚定支持者的不满。知名特斯拉多头、未来基金(Future Fund)管理合伙人加里·布莱克发帖称,降价是一种“短期麻醉剂”,会让潜在客户等待进一步降价。
据外媒9月2日报道,荷兰半导体设备制造商阿斯麦称,尽管荷兰政府颁布的半导体设备出口管制新规9月正式生效,但该公司已获得在2023年底以前向中国运送受限制芯片制造机器的许可。
近日,根据美国证券交易委员会的文件显示,苹果卫星服务提供商 Globalstar 近期向马斯克旗下的 SpaceX 支付 6400 万美元(约 4.65 亿元人民币)。用于在 2023-2025 年期间,发射卫星,进一步扩展苹果 iPhone 系列的 SOS 卫星服务。
据报道,马斯克旗下社交平台𝕏(推特)日前调整了隐私政策,允许 𝕏 使用用户发布的信息来训练其人工智能(AI)模型。新的隐私政策将于 9 月 29 日生效。新政策规定,𝕏可能会使用所收集到的平台信息和公开可用的信息,来帮助训练 𝕏 的机器学习或人工智能模型。
9月2日,荣耀CEO赵明在采访中谈及华为手机回归时表示,替老同事们高兴,觉得手机行业,由于华为的回归,让竞争充满了更多的可能性和更多的魅力,对行业来说也是件好事。
《自然》30日发表的一篇论文报道了一个名为Swift的人工智能(AI)系统,该系统驾驶无人机的能力可在真实世界中一对一冠军赛里战胜人类对手。
近日,非营利组织纽约真菌学会(NYMS)发出警告,表示亚马逊为代表的电商平台上,充斥着各种AI生成的蘑菇觅食科普书籍,其中存在诸多错误。
社交媒体平台𝕏(原推特)新隐私政策提到:“在您同意的情况下,我们可能出于安全、安保和身份识别目的收集和使用您的生物识别信息。”
2023年德国柏林消费电子展上,各大企业都带来了最新的理念和产品,而高端化、本土化的中国产品正在不断吸引欧洲等国际市场的目光。
罗永浩日前在直播中吐槽苹果即将推出的 iPhone 新品,具体内容为:“以我对我‘子公司’的了解,我认为 iPhone 15 跟 iPhone 14 不会有什么区别的,除了序(列)号变了,这个‘不要脸’的东西,这个‘臭厨子’。