简单而面试中又常见的知识点:JS执行机制
- console.log('1');
- setTimeout(function() {
- console.log('2');
- process.nextTick(function() {
- console.log('3');
- })
- new Promise(function(resolve) {
- console.log('4');
- resolve();
- }).then(function() {
- console.log('5')
- })
- });
- process.nextTick(function() {
- console.log('6');
- })
- new Promise(function(resolve) {
- console.log('7');
- resolve();
- }).then(function() {
- console.log('8')
- });
- setTimeout(function() {
- console.log('9');
- process.nextTick(function() {
- console.log('10');
- })
- new Promise(function(resolve) {
- console.log('11');
- resolve();
- }).then(function() {
- console.log('12')
- })
- });
- console.log('程序开始执行~');
- setTimeout(() => {
- console.log('执行setTimeout~');
- }, 1000);
- console.log('程序执行结束~');
- // 输出结果:
- // 程序开始执行~
- // 程序执行结束~
- // ...1s(这里表示等待时间)
- // 执行setTimeout~
- console.log('程序执行开始~')
- setTimeout(() => {
- console.log('setTimeout执行啦~')
- }, 3000);
- sleep(5000);
- console.log('程序执行结束~')
- // 注:这里的sleep函数不是js的标准函数,只是表示一个执行需要5秒的函数。
- // 输出结果:
- // 程序执行开始~
- // ...5s后
- // 程序执行结束~
- // setTimeout执行啦~
- process.nextTick,我们知道浏览器环境下的setTimeout,那么process.nextTick就相当于在node环境下执行的setTimeout。
- 宏任务和微任务,主线程一直在执行script代码,还有setTimeout、setInterval函数就是宏任务,而Promise.then,process.nextTick则是微任务。
- console.log('程序执行开始~')
- setTimeout(() => {
- console.log('setTimeout执行啦~')
- }, 3000);
- new Promise((resolve) => {
- console.log('promise开始执行~');
- resolve();
- }).then(() => {
- console.log('promise执行结束~')
- });
- console.log('程序执行结束~')
- // 输出结果:
- // 程序执行开始~
- // promise开始执行~
- // 程序执行结束~
- // promise执行结束~
- // ...3s后
- // setTimeout执行啦~
- 代码一开始执行,执行的就是全局代码,也就是宏任务的同步代码;
- 遇到console.log,直接执行,输出"程序执行开始~";
- 接着执行,遇到setTimeout函数,将其回调函数注册进宏任务的Event Queue(注意:宏任务和微任务分别有自己的Event Queue);
- 接着遇到new Promise,立刻执行(new Promise里面的函数是立刻执行的,只有.then函数里面才是放到微任务去执行的,不要搞混咯~),输出"promise开始执行~";
- 接着遇到promise.then函数,将其回调函数注册到微任务的Event Queue;
- 接着继续执行,遇到console.log,直接输出"程序执行结束~"
- 到这里,宏任务的同步代码就全部执行完毕了,这时候,js引擎会去检查微任务的Event Queue中是否存在回调函数,这时微任务的Queue中还有一个函数未执行,因此在这时候执行,输出"promise执行结束~";
- 当微任务的所有回调函数被执行完了之后,一次事件循环就结束了。
- 这时候js引擎会检查宏任务的Event Queue中是否还有未执行的函数,如果还有,将会开启下一轮的事件循环。由于此时我们宏任务的Event Queue中还有未执行的setTimeout,所以开启下一轮事件循环,执行setTimeout回调,输出"setTimeout执行啦~"
- console.log('1')
- setTimeout(() => {
- console.log('2')
- })
- new Promise((resolve) => {
- console.log('3')
- resolve()
- }).then(() => {
- console.log('4')
- })
- process.nextTick(() => {
- console.log('5')
- })
- new Promise((resolve) => {
- console.log('6')
- resolve()
- }).then(() => {
- console.log('7')
- })
- process.nextTick(() => {
- console.log('8')
- })
- console.log('9')
- // 输出结果
- // 1 3 6 9 5 8 4 7 2
- console.log('1');
- setTimeout(() => {
- console.log('2');
- process.nextTick(() => {
- console.log('3');
- })
- new Promise((resolve) => {
- console.log('4');
- resolve();
- }).then(() => {
- console.log('5')
- })
- });
- process.nextTick(() => {
- console.log('6');
- })
- new Promise((resolve) => {
- console.log('7');
- resolve();
- }).then(() => {
- console.log('8')
- });
- setTimeout(() => {
- console.log('9');
- process.nextTick(() => {
- console.log('10');
- })
- new Promise((resolve) => {
- console.log('11');
- resolve();
- }).then(() => {
- console.log('12')
- })
- });
- 程序开始,执行宏任务同步代码,遇到console.log,输出:;
- 遇到setTimeout1,将其放入宏任务Event Queue中;
- 遇到process.nextTick1,放入微任务Event Queue中;
- 遇到new Promise,直接执行其中的代码,输出:;
- 遇到Promise.then1函数,将其放入微任务Event Queue;
- 继续执行,遇到setTimeout2,放入宏任务Event Queue;
- 此时任务队列状态:
- 宏Queue: setTimeout1,setTimeout2;
- 微Queue: process.nextTick1、Promise.then1;
- 至此,宏任务同步代码执行完毕,检测微任务队列是否存在任务,由于存在两个微任务,所以这时候执行微任务;
- 先执行process.nextTick1,输出:;
- 接着执行Promise.then1,输出: ;
- 微任务执行完毕后,一次事件循环结束,js引擎持续检测宏任务中是否存在任务,存在的话开启下一次事件循环;由于存在两个setTimeout,所以在满足setTimeout执行条件后,开启下一次事件循环,执行回调函数;
- 先执行setTimeout1,遇到console.log,输出:;
- 接着遇到process.nextTick2,放入微任务Event Queue;
- 继续执行遇到new Promise,直接执行,输出:;
- 然后遇到Promise.then2,放入微任务Event Queue;
- 至此setTimeout1执行完毕,此时任务队列状态:
- 宏Queue: setTimeout2;
- 微Queue: process.nextTick2、Promise.then2;
- js引擎检查微任务Event Queue中还存在两个微任务,因此执行这两个微任务;
- 先执行process.nextTick2,输出:;
- 接着执行Promise.then2,输出:;
- 微任务执行完毕,第二次事件循环结束;
- js引擎持续检查宏任务Event Queue中是否还有未执行函数,检测到还有setTimeout2未执行,因此开启第三轮的事件循环;
- 执行setTimeout2,遇到console.log,输出:;
- 又遇到process.nextTick3,放入微任务队列;
- 遇到new Promise,直接执行,输出:;
- 遇到Promise.then3,放入微任务队列;
- 至此,setTimeout2执行完毕,此时任务队列状态:
- 宏Queue: 无;
- 微Queue: process.nextTick3、Promise.then3;
- js引擎在检测是否存在未执行的微任务,由于还有两个微任务未执行,因此将其执行;
- 先执行process.nextTick3,输出:10;
- 接着执行Promise.then3,输出:12;
- 至此,微任务执行完毕,事件循环结束;
简单而面试中又常见的知识点:JS执行机制的更多相关文章
- 从一道看似简单的面试题重新理解JS执行机制与定时器
壹 ❀ 引 最近在看前端进阶的系列专栏,碰巧看到了几篇关于JS事件执行机制的面试文章,因为我在之前一篇 JS执行机制详解,定时器时间间隔的真正含义 博文中也有记录JS执行机制,所以正好用于作为测试自 ...
- 浏览器中js执行机制学习笔记
浏览器中js执行机制学习笔记 RiverSouthMan关注 0.0772019.05.15 20:56:37字数 872阅读 291 同步任务 当一个脚本第一次执行的时候,js引擎会解析这段代码,并 ...
- 前端js面试中的常见的算法问题
虽说我们很多时候前端很少有机会接触到算法.大多都交互性的操作,然而从各大公司面试来看,算法依旧是考察的一方面.实际上学习数据结构与算法对于工程师去理解和分析问题都是有帮助的.如果将来当我们面对较为复杂 ...
- Nginx面试中最常见的18道题 抱佛脚必备
Nginx的并发能力在同类型网页服务器中的表现,相对而言是比较好的,因此受到了很多企业的青睐,我国使用Nginx网站的知名用户包括腾讯.淘宝.百度.京东.新浪.网易等等.Nginx是网页服务器运维人员 ...
- nginx面试中最常见的18道题
1.请解释一下什么是Nginx? Nginx是一个web服务器和反向代理服务器,用于HTTP.HTTPS.SMTP.POP3和IMAP协议. 2.请列举Nginx的一些特性. Nginx服务器的特性包 ...
- 面试中linux常见的20个命令
1.查找文件 find / -name filename.txt 根据名称查找/目录下的filename.txt文件. 2.查看一个程序是否运行 ps –ef|grep tomcat 查看所有有关to ...
- 企业面试中关于MYSQL重点的28道面试题解答
问题1:char.varchar的区别是什么? varchar是变长而char的长度是固定的.如果你的内容是固定大小的,你会得到更好的性能. 问题2: TRUNCATE和DELETE的区别是什么? ...
- 深入理解 JS 引擎执行机制(同步执行、异步执行以及同步中的异步执行)
首先明确两点: 1.JS 执行机制是单线程. 2.JS的Event loop是JS的执行机制,深入了解Event loop,就等于深入了解JS引擎的执行. 单线程执行带来什么问题? 在JS执行中都是单 ...
- 带你全面了解高级 Java 面试中需要掌握的 JVM 知识点
目录 JVM 内存划分与内存溢出异常 垃圾回收算法与收集器 虚拟机中的类加载机制 Java 内存模型与线程 虚拟机性能监控与故障处理工具 参考 带你全面了解高级 Java 面试中需要掌握的 JVM 知 ...
随机推荐
- tp3.2的__construct和_initialize方法
在tp3.2框架里面,有一个php自带的__construct()构造函数和tp3自带的构造函数_initialize()的实行顺序是先实行 php自带的__construct()构造函数 再实行 t ...
- 安装 wbemcli
安装环境 建立自己的目录后, wget http://vault.centos.org/6.0/os/x86_64/Packages/sblim-wbemcli- ...
- Qt 正则表达式检查 IP 格式
KillerSmath 2018年6月29日 下午10:41 @Pranit-Patil Hi there. Like @jonB says above, you should to replace\ ...
- Django中search fields报错:related Field has invalid lookup: icontains
models.py 文件 # coding:utf8from django.db import models class Book(models.Model): name = model ...
- Omnicore RPC API中文文档
2019独角兽企业重金招聘Python工程师标准>>> OmniCore是比特币核心的一个分支,它在比特币协议之上实现了一个新的Omni协议层,用于代币发行.众售等应用,USDT就是 ...
- python练习---博客登录(装饰器)
程序要求: 1),启动程序,首页面应该显示成如下格式: 欢迎来到博客园首页 1:请登录 2:请注册 3:文章页面 4:日记页面 5:评论页面 6:收藏页面 7:注销 8:退出程序2),用户输入选项,3 ...
- CF思维联系– Codeforces-989C C. A Mist of Florescence
ACM思维题训练集合 C. A Mist of Florescence time limit per test 1 second memory limit per test 256 megabytes ...
- POJ 2955 区间DP必看的括号匹配问题,经典例题
Brackets Time Limit: 1000MS Memory Limit: 65536K Total Submissions: 14226 Accepted: 7476 Description ...
- 洛谷2014 选课(树形DP)树形背包问题
题目描述 在大学里每个学生,为了达到一定的学分,必须从很多课程里选择一些课程来学习,在课程里有些课程必须在某些课程之前学习,如高等数学总是在其它课程之前学习.现在有N门功课,每门课有个学分,每门课有一 ...
- KMP+Tire树(模板)
\(\color{Red}{KMP板子}\) #include <bits/stdc++.h> using namespace std; const int maxn=1e6+9; int ...