细品 JavaScript前端面试题

JavaScript 事件循环(含宏任务与微任务)

2019-10-04  本文已影响0人  越前君

本文更新时间:2019-10-04

JavaScript 特点

JavaScript 是单线程非阻塞的一门语言。单线程意味着:JavaScript 代码在执行的时候只有一个主线程去处理所有的任务,即同一时间只能做一件事情。非阻塞则表示:当执行到一项异步任务的时候,主线程会挂起当前这个异步任务,然后在异步任务返回结果的时候再跟进一定的规则去执行相应的回调。

思考: 为什么 JavaScript 要设计成单线程?
单线程是必要的,也是 Javascript 这门语言的基石,原因之一在其最初也是最主要的执行环境——浏览器中,我们需要进行各种各样的 DOM 操作。试想一下 如果 Javascript 是多线程的,那么当两个线程同时对 DOM 进行一项操作,例如一个向其添加事件,而另一个删除了这个 DOM,此时该如何处理呢?因此,为了保证不会 发生类似于这个例子中的情景,Javascript 选择只用一个主线程来执行代码,这样就保证了程序执行的一致性。

事件循环(Event Loop)

JavaScript 如何实现非阻塞呢?没错,就是通过事件循环的实现的。而事件循环是通过任务队列机制协调的。

在事件循环中,每进行一次循环操作称为 tick,每一次 tick 的任务处理是比较复杂的,主要步骤如下:

tick 需要了解的是:

任务有同步任务、异步任务,而异步任务又分为宏任务(macro-task)微任务(micro-task)

宏任务(macro-task)
可以理解成:每次执行栈的代码就是一个宏任务(包括每次从事件队列中获取的一个事件回调并放到执行中执行)。
浏览器为了能够使得 JS 内部 macro-task 与 DOM 任务有序地执行,会在宏任务执行结束之后,在下一个宏任务开始执行之前,对页面进行重新渲染。

macro-task -> render -> macro-task -> ...

宏任务包括:
setInterval
setTimeout
setImmediate(node.js)
XHR 回调
事件回调(鼠标键盘事件)
indexedDB 数据库等 I/O 操作
UI rendering

微任务(micro-task)
可以理解成:当前 task 执行结束之后立即执行的任务。(在下一个 task 之前 ,在渲染之前)。

macro-task -> micro-task -> render -> macro-task -> ...

微任务包括:
Promise.then catch finally
process.nextTick(node.js)
MutationObserver
Object.observe(已被弃用)

运行机制

参考:

  1. JS事件循环
  2. 详解JavaScript中的Event Loop(事件循环)机制(含node环境下的事件循环机制)
  3. js 宏任务和微任务
上一篇下一篇

猜你喜欢

热点阅读