最近又碰到了event loop问题,之前研究的实在是浅显(https://www.cnblogs.com/zx0423/p/12641637.html)所以今天主要讲述promise的链式调用,async/await,requestAnimationFrame以及MutationObserver()。

1、Promise链式调用

    new Promise((resolve,reject)=>{
console.log("promise1");
resolve();
}).then(()=>{
//callback1
console.log("then11");
new Promise((resolve,reject)=>{
console.log("promise2");
resolve();
}).then(()=>{
// callback2
console.log("then21");
}).then(()=>{
// callback4
console.log("then23");
})
}).then(()=>{
// callback3
console.log("then12");
})

/*
1、new promsie是同步代码,立即执行
2、外部promise第一个then需要等外部promise被resolve后才执行,外部promise立即被resolve,于是执行注册的 then回调。此时外部第二then并未执行,因为他等待的是外部第一个then返回的promise
3、进入外部第一个then,回调里面是一个promise,执行内部promise时,同样两个then链式调用,内部第一个then执 行后,内部promise算执行完,返回一个新的promise给内部第二then用
4、外部第一个then执行完且有返回值,进入外部第二then
5、之后执行内部第二then
*/

解释:

// 行2 ,执行同步的executor函数
栈【 promise 】
宏任务队列【】
微任务队列【 callback1 】
执行结果:promise1
// 行7
栈【 callback1 】
宏任务队列【】
微任务队列【 callback2,callback3 】
执行结果:promise1 , then11 , promise2
//
栈【 callback2,callback3 】
宏任务队列【】
微任务队列【 callback4 】
执行结果:promise1 , then11 , promise2 , then21 , then12 ,
// 行16
栈【 callback4 】
宏任务队列【】
微任务队列【 】
执行结果:promise1 , then11 , promise2 , then21 , then12 , then23

️ 当行8添加 return 语句的时候,答案就截然不同了,大家可以自行体会一下。

2、async/await

  • async函数

  • 返回一个 Promise 对象

     async function test() {
    return 1; // async的函数 在这里隐式 使用 Promise.resolve(1)
    }

    //===>等价于
    new Promise(function(resolve, reject) {
    resolve(1);
    })
  • await

    每次我们使用 await, 解释器都创建一个 promise 对象,然后把剩下的 async 函数中的操作放到 then 回调函数中。async/await 的实现,离不开 Promise。

    1-await后面不是Promise对象,即是一些直接量 / 数值之类的
    • await会阻塞后面的代码,

      • 从右到左执行,

      • 再执行async外面的同步代码,

      • 同步代码执行完,再回到async内部,

      • 把这个非promise的东西,作为 await表达式的结果。

    2-await后面是Promise对象
    • await 它会“阻塞”后面的代码,

      • 从右到左执行,

      • 再执行async外面的同步代码,

      • 【期间不管宏任务微任务,就一直执行,直到...】

      • 直到 Promise 对象 resolved(也就是要等到了结果),即把 resolve 的 参数/值 作为 await 表达式的运算结果。

    理清上述执行顺序,对下面这道题就可以迎刃而解了。

 // async/await 与 then链式调用
async function f1(){
console.log("1");
await 'await的结果'
console.log("3")
} f1(); new Promise(function(resolve){
console.log("2")
resolve();
}).then(function(result){
new Promise( r => r()).then(() =>{
console.log(4)
}).then(()=>{
console.log(5);
})
}) 结果: 1,2,3,4,5

3、requestAnimationFrame & MutationObserver()

基本概念: Window.requestAnimationFrame(callback):告诉浏览器——希望执行一个动画,并且要求浏览器在下次重绘之前调用指定的回调函数更新动画;
      MutationObserver():会在指定的DOM发生变化时被调用。【微任务,很好理解,按照正常的执行顺序执行即可】

大家先来看一段代码吧,思考一下他的执行结果是什么?然后在刷新看看,结果还是不是一样的了?

<div id="testElement"></div>
 setTimeout(() => {
console.log(performance.now(), 'settimeout');
}, 0) requestAnimationFrame(() => {
   console.log(performance.now(),'requestAnimationFrame');
}) const testElement = document.getElementById('testElement'); var observer = new MutationObserver(() => {
console.log('MutationObserver');
}); observer.observe(testElement, {
childList: true
})
//添加子节点,触发MutationObserver()
const div = document.createElement('div');
testElement.appendChild(div); new Promise(resolve => {
console.log('promise');
resolve();
}).then(() => console.log('then')) console.log(performance.now(), 'global');

如图:

可以看到,requestAnimationFrame的执行顺序是不一致的,

  起初,我将requestAnimationFrame归到宏任务中,原因是它绝大多数都会在setTimeout回调执行之后才执行。并将这个结果解释为是由于浏览器在执行渲染的时候,每次执行的时间会有差异,所以导致requestAnimationFrame和setTimeout被压入调用栈的时机不一致,也就导致了回调的时间不一致。

但是,可以看到宏任务与raf任务有明显的差异:
  执行时机不同;
  raf任务队列被执行时,会将其此刻队列中所有任务都执行完,既不属于宏任务/微任务,同时还排在微任务之后。

而且,在一个loop中,可能并不会执行这一步,只要浏览器任务不需要执行渲染,就会跳过。所以会出现requestAnimationFrame顺序不一致的状况。

EventLoop-浏览器篇2的更多相关文章

  1. 前端面试总结——http、html和浏览器篇

    1.http和https https的SSL加密是在传输层实现的. (1)http和https的基本概念 http: 超文本传输协议,是互联网上应用最为广泛的一种网络协议,是一个客户端和服务器端请求和 ...

  2. 浏览器事件循环 & nodejs事件循环

    第1篇:如何理解EventLoop——宏任务和微任务篇 宏任务(MacroTask)引入 在 JS 中,大部分的任务都是在主线程上执行,常见的任务有: 渲染事件 用户交互事件 js脚本执行 网络请求. ...

  3. 如何理解EventLoop--浏览器篇

    前言 最近在准备春招,刷到了JS中的主要运行机制--Event Loop,觉得它的实现思路有必要整理一下,以防忘记.关于它在浏览器上的实现,我结合了自己的理解以及示例代码,想用最通俗的语言表达出来.如 ...

  4. 浏览器如何执行JS

    作为JS系列的第一篇,内容当然是浏览器如何执行一段JS啦. 首先通过浏览器篇我们可以得知,JS是在渲染进程里的JS引擎线程执行的.在此之后还要了解几个概念,编译器(Compiler).解释器(Inte ...

  5. Event Loop

    Event Loop 是 JavaScript 异步编程的核心思想,也是前端进阶必须跨越的一关.同时,它又是面试的必考点,特别是在 Promise 出现之后,各种各样的面试题层出不穷,花样百出.这篇文 ...

  6. 总结:JavaScript异步、事件循环与消息队列、微任务与宏任务

    本人正在努力学习前端,内容仅供参考.由于各种原因(不喜欢博客园的UI),大家可以移步我的github阅读体验更佳:传送门,喜欢就点个star咯,或者我的博客:https://blog.tangzhen ...

  7. (转)总结:JavaScript异步、事件循环与消息队列、微任务与宏任务

    前言 Philip Roberts 在演讲 great talk at JSConf on the event loop 中说:要是用一句话来形容 JavaScript,我可能会这样: “JavaSc ...

  8. html5手机常见问题与工具分享

    mobileTech A useful tools or tips list for mobile web application developing 这个项目收集移动端开发所需要的一些资源与小技巧 ...

  9. mobileTech

    A useful tools or tips list for mobile web application developing 这个项目收集移动端开发所需要的一些资源与小技巧 工具类网站 HTML ...

  10. TGL站长关于常见问题的回复

    问题地址: http://www.thegrouplet.com/thread-112923-1-1.html 问题: 网站配有太多的模板是否影响网站加载速度 月光答复: wp不需要删除其他的模板,不 ...

随机推荐

  1. PHP children() 函数

    实例 查找 note 节点的子节点: <?php$note=<<<XML<note><to>Tove</to>高佣联盟 www.cgewan ...

  2. Skill 导出所有Layer信息用于tapeout

    https://www.cnblogs.com/yeungchie/ 用于在 tapeout 前要走的流程,foundry 会需要你上传一份芯片用到的所有 Layer 的 excel 文档. TAB ...

  3. luogu P2510 [HAOI2008]下落的圆盘

    LINK:下落的圆盘 计算几何.n个圆在平面上编号大的圆将编号小的圆覆盖求最后所有没有被覆盖的圆的边缘的总长度. 在做这道题之前有几个前置知识. 极坐标系:在平面内 由极点 极轴 和 极径组成的坐标系 ...

  4. 实战:docker搭建FastDFS文件系统并集成SpringBoot

    实战:docker搭建FastDFS文件系统并集成SpringBoot 前言 15年的时候,那时候云存储还远远没有现在使用的这么广泛,归根结底就是成本和安全问题,记得那时候我待的公司是做建站开发的,前 ...

  5. day8.列表、字典、集合相关操作

    一.列表的相关操作 # (1)列表的拼接 (同元组) # (2)列表的重复 (同元组) # (3)列表的切片 (同元组) # 语法 => 列表[::] 完整格式:[开始索引:结束索引:间隔值] ...

  6. Newbe.Claptrap 框架如何实现多级生命周期控制?

    Newbe.Claptrap 框架如何实现多级生命周期控制?最近整理了一下项目的术语表.今天就谈谈什么是 Claptrap Lifetime Scope. 特别感谢 kotone 为本文提供的校对建议 ...

  7. JSON 和 POJO 互转,List<T> 和 JSON 互转

    JSON 和 POJO import com.alibaba.fastjson.JSONObject; import org.slf4j.Logger; import org.slf4j.Logger ...

  8. Docker 启动 OpenResty

    Docker 启动 OpenResty OpenResty 是一个基于 Nginx 与 Lua 的高性能 Web 平台,其内部集成了大量精良的 Lua 库.第三方模块以及大多数的依赖项.用于方便地搭建 ...

  9. Ambiguous mapping. Cannot map 'xxxController' method

    @GetMapping public JsonResp<List<DtoLandRegion>> getLandRegionList() { List<DtoLandRe ...

  10. Java程序员面试必备:Volatile全方位解析

    前言 volatile是Java程序员必备的基础,也是面试官非常喜欢问的一个话题,本文跟大家一起开启vlatile学习之旅,如果有不正确的地方,也麻烦大家指出哈,一起相互学习~ 1.volatile的 ...