实例分析js事件循环机制


Posted in Javascript onDecember 13, 2017

本文通过实例给大家详细分析了JS中事件循环机制的原理和用法,以下是全部内容:

var start = new Date()
setTimeout(function () {
 var end = new Date
 console.log('Time elapsed:', end - start, 'ms')
}, 500)
while (new Date() - start < 1000) {
}

有其他语言能完成预期的功能吗?Java, 在Java.util.Timer中,对于定时任务的解决方案是通过多线程手段实现的,任务对象存储在任务队列,由专门的调度线程,在新的子线程中完成任务的执行

js是单线程的

JavaScript的主要用途是与用户互动,以及操作DOM。这决定了它只能是单线程,否则会带来很复杂的同步问题。

为了利用多核CPU的计算能力,HTML5提出Web Worker标准,允许JavaScript脚本创建多个线程,但是子线程完全受主线程控制,且不得操作DOM。所以,这个新标准并没有改变JavaScript单线程的本质。

函数调用栈和任务队列

实例分析js事件循环机制

调用栈

JS执行时会形成调用栈,调用一个函数时,返回地址、参数、本地变量都会被推入栈中,如果当前正在运行的函数中调用另外一个函数,则该函数相关内容也会被推入栈顶.该函数执行完毕,则会被弹出调用栈.变量也随之弹出,由于复杂类型值存放于堆中,因此弹出的只是指针,他们的值依然在堆中,由GC决定回收.

事件循环(event loop) & 任务队列(task queue)

JavaScript 主线程拥有一个执行栈以及一个任务队列

遇到异步操作(例如:setTimeout, AJAX)时,异步操作会由浏览器(OS)执行,浏览器会在这些任务完成后,将事先定义的回调函数推入主线程的任务队列(task queue)中,当主线程的执行栈清空之后会读取task queue中的回调函数,当task queue被读取完毕之后,主线程接着执行,从而进入一个无限的循环,这就是事件循环.

主线程执行栈 & 任务队列 循环执行,构成事件循环

结论

setTimeout()只是将事件插入了"任务队列",必须等到当前代码(执行栈)执行完,主线程才会去执行它指定的回调函数。要是当前代码耗时很长,有可能要等很久,所以并没有办法保证,回调函数一定会在setTimeout()指定的时间执行。

另一个例子

(function test() {
 setTimeout(function() {console.log(4)}, 0);
 new Promise(function executor(resolve) {
 console.log(1);
 for( var i=0 ; i<10000 ; i++ ) {
 i == 9999 && resolve();
 }
 console.log(2);
 }).then(function() {
 console.log(5);
 });
 console.log(3);
})()

Macrotask & Microtask

macrotask 和 microtask 是异步任务的两种分类。在挂起任务时,JS 引擎会将所有任务按照类别分到这两个队列中,首先在 macrotask 的队列(这个队列也被叫做 task queue)中取出第一个任务,执行完毕后取出 microtask 队列中的所有任务顺序执行;之后再取 macrotask 任务,周而复始,直至两个队列的任务都取完。

macro-task: script(整体代码), setTimeout, setInterval, setImmediate, I/O, UI rendering
micro-task: process.nextTick, Promises(这里指浏览器实现的原生 Promise), Object.observe, MutationObserver

实例分析js事件循环机制

结论

全部代码(script) macrotask -> microtask queue (含有promise.then) -> macrotask(setTimeout) -> 下一个microtask

Node.js的事件循环

process.nextTick & setImmediate

process.nextTick指定的任务总是发生在所有异步任务之前

setImmediate指定的任务总是在下一次Event Loop时执行

process.nextTick(function A() {
 console.log(1);
 process.nextTick(function B(){console.log(2);});
});
setTimeout(function timeout() {
 console.log('TIMEOUT FIRED');
}, 0)
new Promise(function(resolve) {
 console.log('glob1_promise');
 resolve();
}).then(function() {
 console.log('glob1_then')
})
process.nextTick(function() {
 console.log('glob1_nextTick');
})
Javascript 相关文章推荐
jquery 插件实现图片延迟加载效果代码
Feb 06 Javascript
JavaScript插件化开发教程 (四)
Jan 27 Javascript
jQuery实现的可编辑表格完整实例
Jun 20 Javascript
js判断iframe中元素是否存在的实现代码
Dec 24 Javascript
ionic 3.0+ 项目搭建运行环境的教程
Aug 09 Javascript
React Native 集成jpush-react-native的示例代码
Aug 16 Javascript
浅谈Angular6的服务和依赖注入
Jun 27 Javascript
在vue中安装使用vux的教程详解
Sep 16 Javascript
JS 正则表达式验证密码、邮箱格式的实例代码
Oct 28 Javascript
Vue.js组件高级特性实例详解
Dec 24 Javascript
jquery实现抽奖功能
Oct 22 jQuery
HTML+VUE分页实现炫酷物联网大屏功能
May 27 Vue.js
javascript实现QQ空间相册展示源码
Dec 12 #Javascript
自定义PC微信扫码登录样式写法
Dec 12 #Javascript
基于模板引擎Jade的应用(详解)
Dec 12 #Javascript
jquery获取transform里的值实现方法
Dec 12 #jQuery
JS排序算法之希尔排序与快速排序实现方法
Dec 12 #Javascript
将Sublime Text 3 添加到右键中的简单方法
Dec 12 #Javascript
详解vue渲染函数render的使用
Dec 12 #Javascript
You might like
PHP 常用函数库和一些实用小技巧
2009/01/01 PHP
PHP 第二节 数据类型之转换
2012/04/28 PHP
PHP类继承 extends使用介绍
2014/01/14 PHP
php字符串操作针对负值的判断分析
2016/07/28 PHP
PHP实现小程序批量通知推送
2018/11/27 PHP
js获取单选按钮的数据
2006/11/27 Javascript
在JavaScript中,为什么要尽可能使用局部变量?
2009/04/06 Javascript
js 屏蔽鼠标右键脚本附破解方法
2009/12/03 Javascript
一次失败的jQuery优化尝试小结
2011/02/06 Javascript
javascript学习笔记(七) js函数介绍
2012/06/19 Javascript
javascript动态加载二
2012/08/22 Javascript
js实现动态加载脚本的方法实例汇总
2015/11/02 Javascript
通过隐藏iframe实现无刷新上传文件操作
2016/03/16 Javascript
引入JavaScript时alert弹出框显示中文乱码问题
2017/09/16 Javascript
angular2 组件之间通过service互相传递的实例
2018/09/30 Javascript
浅谈在Vue.js中如何实现时间转换指令
2019/01/06 Javascript
javascript function(函数类型)使用与注意事项小结
2019/06/10 Javascript
关于layui flow loading占位图的实现方法
2019/09/21 Javascript
[03:12]2016完美“圣”典风云人物:单车专访
2016/12/02 DOTA
详解Python给照片换底色(蓝底换红底)
2019/03/22 Python
pyinstaller打包程序exe踩过的坑
2019/11/19 Python
Python实现把类当做字典来访问
2019/12/16 Python
python图形界面开发之wxPython树控件使用方法详解
2020/02/24 Python
Python matplotlib绘制图形实例(包括点,曲线,注释和箭头)
2020/04/17 Python
使用Python-OpenCV消除图像中孤立的小区域操作
2020/07/05 Python
详解css3中dispaly的Grid布局与Flex布局
2020/09/11 HTML / CSS
div或img图片高度随宽度自适应的方法
2020/02/06 HTML / CSS
关于毕业的广播稿
2014/01/10 职场文书
新闻专业毕业生求职信
2014/08/08 职场文书
民政局副局长民主生活会个人整改措施
2014/10/04 职场文书
学前班幼儿评语大全
2014/12/29 职场文书
60条职场经典语录,总有一条能触动你的心
2019/08/21 职场文书
关于拾金不昧的感谢信(五篇)
2019/10/18 职场文书
Spring Data JPA框架持久化存储数据到数据库
2022/04/28 Java/Android
Hive HQL支持2种查询语句风格
2022/06/25 数据库
JS前端使用Canvas快速实现手势解锁特效
2022/09/23 Javascript