博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
Node中的事件循环
阅读量:7071 次
发布时间:2019-06-28

本文共 3159 字,大约阅读时间需要 10 分钟。

事件循环中的各阶段

Node.js 的事件循环流程大致如下:

┌───────────────────────────┐┌─>│           timers          ││  └─────────────┬─────────────┘│  ┌─────────────┴─────────────┐│  │     pending callbacks     ││  └─────────────┬─────────────┘│  ┌─────────────┴─────────────┐│  │       idle, prepare       ││  └─────────────┬─────────────┘      ┌───────────────┐│  ┌─────────────┴─────────────┐      │   incoming:   ││  │           poll            │<─────┤  connections, ││  └─────────────┬─────────────┘      │   data, etc.  ││  ┌─────────────┴─────────────┐      └───────────────┘│  │           check           ││  └─────────────┬─────────────┘│  ┌─────────────┴─────────────┐└──┤      close callbacks      │   └───────────────────────────┘复制代码

每个阶段都有自己的任务队列,当本阶段的任务队列都执行完毕,或者达到了执行的最大任务数,就会进入到下一个阶段。

timers 阶段

这个阶段会执行被 setTimeoutsetInterval 设置的定时任务。 当然,这个定时并不是准确的,而是在超过了定时时间后,一旦得到执行机会,就立刻执行。

pending callbacks 阶段

这个阶段会执行一些和底层系统有关的操作,例如TCP连接返回的错误等。这些错误发生时,会被Node 推迟到下一个循环中执行。

轮询阶段

这个阶段是用来执行和 IO 操作有关的回调的,Node会向操作系统询问是否有新的 IO 事件已经触发,然后会执行响应的事件回调。几乎所有除了 定时器事件、 setImmediate()close callbacks 之外操作都会在这个阶段执行。

check 阶段

这个阶段会执行 setImmediate() 设置的任务。

close callbacks 阶段

如果一个 sockethandle(句柄) 突然被关闭了,例如通过 socket.destroy() 关闭了,close 事件将会在这个阶段发出。

事件循环的具体执行

事件循环初始化之后,会按照上图所示的流程进行:

  1. 首先会依次执行 定时器中的任务、 pending callback 回调;
  2. 然后进入到 idleprepare 阶段,这里会执行 Node 内部的一些逻辑;
  3. 然后进入到 poll 轮询阶段。在这个阶段会执行所有的 IO 回调,如 读取文件,网络操作等。 poll 阶段有一个 poll queue 任务队列。这个阶段的执行过程相对较长,具体如下:
  • 进入到本阶段,会先检查 timeout 定时队列是否有可执行的任务,如果有,会跳转到 定时器阶段 执行。
  • 如果没有 定时器任务 ,就会检查 poll queue 任务队列,如果不为空,会遍历执行所有任务直到都执行完毕或者达到能执行的最大的任务数量。
  • poll queue 任务队列执行完成后,会检查 setImmediate 任务队列是否有任务,如果有的话,事件循环会转移到下一个 check 阶段。
  • 如果没有 setImmediate 任务,那么,Node 将会在此等待,等待新的 IO 回调的到来,并立刻执行他们。 注意 :这个等待不会一直等待下去,而是达到一个限定条件之后,继续转到下一个阶段去执行。

setTimeout()setImmediate()

一个小秘密

其实也不算秘密,只是我是在刚刚查阅资料才知道的。 那就是:在 Node 中,setTimeout(callback, 0) 会被转换为 setTimeout(callback, 1) 。 详情请参考 。

setTimeout()setImmediate() 的执行顺序

下面这两个定时任务执行的顺序在不同情况下,表现不一致。

setTimeout(function() {    console.log('timeout');}, 0);setImmediate(function() {    console.log('immediate');});复制代码

普通代码中设置定时器

如果在普通的代码执行阶段(例如在最外层代码块中),设置这两个定时任务,他们的执行顺序是不固定的。

  1. 首先,我们设置的 setTimeout(callback, 0) 已经被转换成为 setTimeout(callback, 1) ,所以进入 定时器 阶段时,会根据当前时间判断定时是否超过了 1ms
  2. 事件循环在进入定时器阶段之前会由系统调用方法来更新当前时间,由于系统中同时运行着其他的程序,系统需要等待其他程序的进程运行结束才能获取准确时间,所以更新得到的时间可能会有一定的延迟。
  3. 更新时间时,若没有延迟,定时不到 1ms ,immediate 任务会先执行;如果存在延迟,并且这个时间达到了 1ms 的界限, timeout 任务就会首先执行。

在IO回调中设置定时器

如果我们在 IO 回调中设置了这两个定时器,那么 setImmediate 任务会首先执行,原因如下:

  1. 进入 poll phase 轮询阶段之前会先检查是否有 timer 定时任务。
  2. 如果没有 timer 定时任务,才会执行后面的 IO 回调。
  3. 我们在 IO 回调中设置 setTimeout 定时任务,这时已经过了 timer 检查阶段,所以 timer 定时任务会被推迟到下一个循环中执行。

process.nextTick()

无论在事件循环的哪个阶段,只要使用 process.nextTick() 添加了回调任务,Node 都会在进入下一阶段之前把 nextTickQueue 队列中的任务执行完。

setTimeout(function() {    setImmediate(() => {        console.log('immediate');    });    process.nextTick(() => {        console.log('nextTick');    });}, 0);// nextTick// immediate复制代码

上述代码中,总是先执行 nextTick 任务,就是因为在循环在进入下一个阶段之前会先执行 nextTickQueue 中的任务。下面代码的执行结果也符合预期。

setImmediate(() => {    setTimeout(() => {        console.log('timeout');    }, 0);    process.nextTick(() => {        console.log('nextTick');    });});// nextTick// timeout复制代码

转载地址:http://ieell.baihongyu.com/

你可能感兴趣的文章
CDN(内容分发网络)技术原理
查看>>
Flask + mod_wsgi + Apache on Windows 部署成功(随时接受提问)
查看>>
提高代码编码的效率,习惯非常重要!
查看>>
maven最全教程
查看>>
对Inductive Bias(归纳偏置)的理解
查看>>
chest
查看>>
hdu 1215 七夕节
查看>>
老调重弹:JDBC系列 之 &lt;驱动载入原理全面解析&gt;
查看>>
UVa11183 - Teen Girl Squad(最小树形图-裸)
查看>>
高速排序--双边扫描与单边扫描的实现
查看>>
android去除标题栏和状态栏
查看>>
[转]利用 NPOI 變更字體尺寸及樣式
查看>>
eval解析JSON字符串的一个小问题
查看>>
jquery简单原则器(匹配除了指定选择器之外的元素 selector 表示选择器)
查看>>
update使用inner join
查看>>
Vue2.x中的父子组件相互通信
查看>>
多种替身邮方法总结!
查看>>
沟通比文档更有力
查看>>
在页面头部<!DOCTYPE html ....> 前面不能有任何输出
查看>>
hdu 2102 A计划(双层BFS)(具体解释)
查看>>