我的编程空间,编程开发者的网络收藏夹
学习永远不晚

Javascript运行机制之EventLoop

短信预约 -IT技能 免费直播动态提醒
省份

北京

  • 北京
  • 上海
  • 天津
  • 重庆
  • 河北
  • 山东
  • 辽宁
  • 黑龙江
  • 吉林
  • 甘肃
  • 青海
  • 河南
  • 江苏
  • 湖北
  • 湖南
  • 江西
  • 浙江
  • 广东
  • 云南
  • 福建
  • 海南
  • 山西
  • 四川
  • 陕西
  • 贵州
  • 安徽
  • 广西
  • 内蒙
  • 西藏
  • 新疆
  • 宁夏
  • 兵团
手机号立即预约

请填写图片验证码后获取短信验证码

看不清楚,换张图片

免费获取短信验证码

Javascript运行机制之EventLoop

一、四个概念

1、Javascript是单线程

单线程意味着我们的js代码只能从上往下同步执行,同一时间只能执行一个任务,这会导致某些执行时间较长或者执行时间不确定的任务会卡住其它任务的正常执行,Event Loop出现的原因正是为了解决此问题。

2、任务队列

为了解决上述的排队问题,有了任务队列,浏览器在异步任务有了结果后,将其添加到任务队列,以待将来执行,其他任务就在主线程上同步执行。

这里要注意的是,向任务队列中添加任务的时机是异步任务有结果后。其实任务队列中存在的就是异步任务的回调函数。

3、同步任务、异步任务

Js程序中的同步任务是指在主线程中执行的任务,异步任务是指进入任务队列中的任务

4、Javascript执行栈

所有的同步任务都在主线程上执行,行成一个执行栈。当主线程上任务执行完毕后,从任务队列中取出任务执行。


var name = "zhouwei";

setTimeout(() => {
    console.log(1);
}, 1000);

console.log(name);


上面代码在浏览器中执行如下,我们将程序全局执行环境的代码理解为包裹在一个main函数中的代码,这段代码的执行栈变化如下图:

  • 开始执行代码,将main任务(全局代码入栈执行),当遇到异步任务(setTimeout后)。
  • 浏览器接管异步任务,并在1s后将异步任务的结果(回调函数)添加到任务队列。
  • 执行栈中的同步任务执行完毕,此时任务队列为空(未到1s),执行栈也为空
  • 异步任务有结果后,首先进入任务队列排队(因为可能有很多异步任务)。
  • 执行栈从任务队列中取出任务开始同步执行。
  • 重复执行第5步。

二、Event Loop

Js执行栈不断的从任务队列中读取任务并执行的过程就是Event Loop

我们知道任务队列中存放的是异步任务的结果,那么异步任务都有哪些了?

  • 1、事件

Javascript中的事件有很多,都是属于异步任务。由浏览器接管,当事件触发时,将事件的回调加入的任务队列中,在Js执行栈中没有任务时执行。

  • 2、Http请求
  • 3、定时器
  • 4、requestAnimationFrame等

宏任务(macrotask)和微任务(microtask)
在了解了任务队列和Event Loop后,我们知道了Js执行栈从任务队列中读取任务执行,但这个具体工程我们任务不清楚,这里引出了宏任务和微任务的的概念,帮助我们理解Event Loop。

进入任务队列中的异步任务回调分为了宏任务和微任务, Js执行栈执行宏任务和微任务的规则如下图所示。

Js执行栈首先执行一个宏任务(全局代码) -> 从任务队列中读取所有微任务执行 -> UI rendering(浏览器渲染界面) -> 从任务队列读取一个宏任务 -> 所有微任务 -> UI rendering -> …

在每一轮的Event Loop结束后(1个宏任务 + 所有微任务),浏览器开始渲染界面(如果有需要渲染的UI,否则不执行UI rendering),在UI rendering结束后,开始下一轮Event Loop。

哪些是宏任务?

  • setTimeout
  • setInterval
  • setImmediate (Node)
  • requestAnimationFrame (浏览器)
  • I/O (事件回调)
  • UI rendering (浏览器渲染)

哪些是微任务?

  • Promise
  • process.nextTick (Node)
  • MutationObserver (现代浏览器提供的用来检测 DOM 变化的网页接口)

setTimeout延时问题

一般来说在代码中setTimeout中回调的执行时间都是大于设置的时间。 这是因为在setTimeout指定时间到达后,虽然回调函数被添加到了任务队列,但是此时Js执行栈中可能有正在执行的任务,此回调需要等待Js执行栈的任务执行完毕后才有机会执行,这就是setTimeout延时问题。

三、实战

练习下下方代码输出结果吧:


console.log(1);

setTimeout(() => {
    console.log(2);
    Promise.resolve().then(() => {
        console.log(3)
    });
});

new Promise(resolve => {
    console.log(4);
    setTimeout(() => {
        console.log(5);
    });
    resolve(6)
}).then(data => {
    console.log(data);
})

setTimeout(() => {
    console.log(7);
})

console.log(8);

用上方的我们说过的js执行机制来分析这道题:

1: 执行全局任务中的同步代码输出:

1
4
8

这里需要注意的是Promise接受的handle函数是同步任务,而then方法是异步任务,所以会直接输出4。

2: 这时的任务队列中有三个setTimeout的宏任务,和一个Promise的微任务


// 此时的宏任务是

setTimeout(() => {
    console.log(2);
    Promise.resolve().then(() => {
        console.log(3)
    });
});

setTimeout(() => {
    console.log(5);
});


setTimeout(() => {
    console.log(7);
})

// 此时微任务是
then(data => {
    console.log(data);
})

执行一个微任务, 输出:6

3: 接着执行第一个宏任务


setTimeout(() => {
    console.log(2);
    Promise.resolve().then(() => {
        console.log(3)
    });
});


输出:2

在此宏任务中,向任务 队列添加了一个微任务。此时任务队列有了新的微任务。

4:执行一个微任务,输出:3


then(() => {
   console.log(3)
});


5: 继续按照规则执行任务, 输出: 5、7

整体输出情况是:

1、4、8、6、2、3、5、7

你的答案是不是这样呢?

总结:

  • javascritp的任务分为同步任务和异步任务
  • 同步任务在主线程(Js执行栈)中执行,异步任务被其他线程接管,并在异步任务有结果后,将其回调添加到任务队列。
  • 任务队列中的任务分为了宏任务和微任务。Js执行栈总是先执行一个宏任务,再执行完所有微任务…

到此这篇关于Javascript运行机制之Event Loop的文章就介绍到这了,更多相关Javascript运行机制Event Loop内容请搜索编程网以前的文章或继续浏览下面的相关文章希望大家以后多多支持编程网!

免责声明:

① 本站未注明“稿件来源”的信息均来自网络整理。其文字、图片和音视频稿件的所属权归原作者所有。本站收集整理出于非商业性的教育和科研之目的,并不意味着本站赞同其观点或证实其内容的真实性。仅作为临时的测试数据,供内部测试之用。本站并未授权任何人以任何方式主动获取本站任何信息。

② 本站未注明“稿件来源”的临时测试数据将在测试完成后最终做删除处理。有问题或投稿请发送至: 邮箱/279061341@qq.com QQ/279061341

Javascript运行机制之EventLoop

下载Word文档到电脑,方便收藏和打印~

下载Word文档

猜你喜欢

深入阐述JavaScript的运行机制

JavaScript是一种流行的编程语言,它几乎被用于网站和应用程序的开发中。对于初学者和许多开发人员来说,JavaScript似乎是单线程的,这意味着它只能同时执行一个任务。然而,JavaScript到底是不是真的单线程呢?这是一个有争议的话题,这篇文章将探讨这个话题并阐述JavaScript的运行机制。JavaScript是一种脚本语言,用于在网页上实现动态效果。它最初被设
2023-05-14

JS的执行机制(EventLoop、宏任务和微任务)

这篇文章主要介绍了JS的执行机制(EventLoop、宏任务和微任务),具有很好的参考价值,希望对大家有所帮助。
2023-01-28

手机javascript运行错误

近年来,手机已经成为我们日常生活中不可或缺的一部分,而javascript作为现代web开发中最常用的技术之一,也是手机应用开发的重要组成部分。但是,在手机应用开发中,javascript运行错误成为了一个频繁出现的问题,给开发者带来了困扰。本文将探讨手机javascript运行错误的原因,以及如何避免和解决这些问题。一、原因分析1、浏览器兼容问题在手机端,不同的浏览器厂商、版
2023-05-17

JavaScript中的事件循环机制及其运行原理

JavaScript中的事件循环机制是一种异步处理机制,通过维护事件队列和消息队列,实现任务的分发和执行。事件循环机制由主线程和任务队列构成,主线程运行完当前任务后会检查任务队列中是否有待执行的任务,如有则执行,否则等待
2023-05-18

kubernetes实践之五十一:kube-proxy运行机制分析

一: Service,Endpoints与Pod的关系Kube-proxy进程获取每个Service的Endpoints,实现Service的负载均衡功能二:Service的负载均衡转发规则访问Service的请求,不论是Cluster I
2023-06-04

如何进行C++运行机制的介绍

这期内容当中小编将会给大家带来有关如何进行C++运行机制的介绍,文章内容丰富且以专业的角度为大家分析和叙述,阅读完这篇文章希望大家可以有所收获。C++编程语言是一款应用广泛,功能强大的编程应用语言。它支持多重编程范式的通用程序设计语言,并且
2023-06-17

React运行机制超详细讲解

React作为当下最为流行的前端开发框架之一,使用它可以快速构建大型Web应用,加上其出色的性能表现,使得众多互联网公司对它格外地青睐,这篇文章主要介绍了React运行机制
2022-11-13

揭秘Golang中锁的运行机制

Golang中锁的工作原理探究在并发编程中,锁是一种重要的同步机制,用于保护共享资源的访问。Golang通过内置的sync包提供了锁的支持,使得我们可以在多个goroutine之间安全地共享数据。本文将深入探究Golang中锁的工作原理,
揭秘Golang中锁的运行机制
2024-01-24

java运行机制指的是什么

这篇文章给大家分享的是有关java运行机制指的是什么的内容。小编觉得挺实用的,因此分享给大家做个参考,一起跟随小编过来看看吧。Java程序运行时,必须经过编译和运行两个步骤:首先将后缀名为java的源文件进行编译,最终生成后缀名为.clas
2023-06-06

源码解析springbatch的job运行机制

这篇文章主要介绍了springbatch的job是如何运行的,本文给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友可以参考下
2022-11-13

深入理解PHP逻辑运行机制

当我们谈论PHP的逻辑运行机制时,我们实际上是在探讨PHP是如何解释和执行我们写的代码的。PHP是一种服务器端脚本语言,它可以与HTML一起工作,从而创建动态网页。在本文中,我们将深入探讨PHP的逻辑运行机制,并通过具体的代码示例来帮助读者
深入理解PHP逻辑运行机制
2024-03-06

编程热搜

目录