本网站(662p.com)打包出售,且带程序代码数据,662p.com域名,程序内核采用TP框架开发,需要联系扣扣:2360248666 /wx:lianweikj
精品域名一口价出售:1y1m.com(350元) ,6b7b.com(400元) , 5k5j.com(380元) , yayj.com(1800元), jiongzhun.com(1000元) , niuzen.com(2800元) , zennei.com(5000元)
需要联系扣扣:2360248666 /wx:lianweikj
一篇文章带你搞懂JavaScript 微任务(Microtask)
BilyLiang · 196浏览 · 发布于2021-08-18 +关注

本文基于JavaScript基础,介绍了微任务。其中.then/catch/finally 处理程序(handler),总是在当前代码完成后才会被调用。


大家好,我进阶学习者。

一、前言

Promise 处理始终是异步的,因为所有 promise 行为都会通过内部的 “promise jobs” 队列,也被称为“微任务队列”。

Promise 的处理程序(handlers).then、.catch 和 .finally 都是异步的。

即便一个 promise 立即被 resolve,.then、.catch 和 .finally ,下面的代码也会在这些处理程序(handler)之前被执行。

代码如下:


  1. let promise = Promise.resolve(); 

  2. promise.then(() => alert("promise done!")); 

  3. alert("code finished");  

  4. // 这个 alert 先显示 

如果运行它,会首先看到 code finished,然后才是 promise done。这很奇怪,因为这个 promise 肯定是一开始就完成的。

运行结果:

为什么 .then 会在之后才被触发?这是怎么回事?

二、微任务队列(Microtask queue)

1. 如果执行顺序对很重要该怎么办?

Promise 的处理程序(handler)总是会经过这个内部队列。

如果有一个包含多个 .then/catch/finally 的链,那么它们中的每一个都是异步执行的。也就是说,它会首先进入队列。

然后在当前代码执行完成并且先前排队的处理程序(handler)都完成时才会被执行。

2. 怎么才能让 code finished 在 promise done 之后运行呢?

很简单,只需要像下面这样使用 .then 将其放入队列:


  1. Promise.resolve() 

  2.   .then(() => alert("promise done!")); //规定相对应的顺序 

  3.   .then(() => alert("code finished")); 

上面代码,加上这语句就是按照预期执行的。

三、未处理的 rejection

现在,可以确切地看到 JavaScript 是如何发现未处理的 rejection 的。

如果一个 promise 的 error 未被在微*任务*队列的末尾进行处理,则会出现“未处理的 rejection”。

正常来说。

如果预期可能会发生错误,会在 promise 链上添加 .catch 来处理 error:


  1. let promise = Promise.reject(new Error("Promise Failed!")); 

  2. promise.catch(err => alert('caught')); 

  3. // 不会运行:error 已经被处理 

  4. window.addEventListener('unhandledrejection', event => alert(event.reason)); 

运行结果:

但是如果忘记添加 .catch,那么,微任务队列清空后,JavaScript 引擎会触发下面这事件:


  1. let promise = Promise.reject(new Error("Promise Failed!")); 

  2. // Promise Failed! 

  3. window.addEventListener('unhandledrejection', event => alert(event.reason)); 

运行结果:

如果迟一点再处理这个 error 会怎样?

例:


  1. let promise = Promise.reject(new Error("Promise Failed!")); 

  2. setTimeout(() => promise.catch(err => alert('caught')), 1000); 

  3. // Error: Promise Failed! 

  4. window.addEventListener('unhandledrejection', event => alert(event.reason)); 

现在,如果运行上面这段代码,会先看到 Promise Failed!,然后才是 caught。

注:

如果并不了解微任务队列,可能会想:“为什么 unhandledrejection 处理程序(handler)会运行?已经捕获(catch)并处理了 error!”。

当微任务队列中的任务都完成时,才会生成 unhandledrejection:引擎会检查 promise,如果 promise 中的任意一个出现 “rejected” 状态,unhandledrejection 事件就会被触发。

在上面这个例子中,被添加到 setTimeout 中的 .catch 也会被触发。只是会在 unhandledrejection 事件出现之后才会被触发。

四、总结

本文基于JavaScript基础,介绍了微任务。其中.then/catch/finally 处理程序(handler),总是在当前代码完成后才会被调用。

如果需要确保一段代码,在 .then/catch/finally 之后被执行,可以将它添加到链式调用的 .then 中。

在大多数 JavaScript 引擎中(包括浏览器和 Node.js),微任务(microtask)的概念与“事件循环(event loop)”和“宏任务(macrotasks)”紧密相关。

代码很简单,希望能够帮助你更好的学习。


相关推荐

RN开发环境的npm私库本地debug调试

manongba · 698浏览 · 2019-05-09 17:03:46
你不知道的浏览器渲染原理

追忆似水年华 · 1373浏览 · 2019-05-09 22:47:56
基于iview的router常用控制方式

追忆似水年华 · 999浏览 · 2019-06-03 10:39:21
编程小知识之 JavaScript 文件读取

manongba · 717浏览 · 2019-06-10 09:16:16
10个省时间的 PyCharm 技巧 赶快收藏!

· 703浏览 · 2019-06-10 09:32:01
加载中

0评论

评论
我是湖南最靓仔的那条仔,希望来到这里能够交道志同道合的朋友,一起学习,不断进步!!!
分类专栏
小鸟云服务器
扫码进入手机网页