zl程序教程

您现在的位置是:首页 >  Javascript

当前栏目

nodejs事件循环

2023-04-18 16:50:38 时间

6个主要阶段

当js同步脚本运行完后,如果有异步操作还没有完成,node就将进入事件循环,像http.createServer.listen,fs.readFileAsync等操作都会使node进入事件循环,没有的话node将直接退出。

timers pending callbacks idle, prepare poll check close callbacks

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

每个阶段都维护一个任务队列,进行下一阶段的最低条件是清空本阶段任务队列的任

重点阶段说明

1.timers

这个阶段执行 setTimeout(callback) 和 setInterval(callback) 预定的 callback; 注意:setTimeout(callback,0) 在node中会被强制转换为setTimeout(callback,1)

2.pending,callback

系统底层操作的回调,如TCP错误

3.idle,prepare

闲置预备阶段,仅供内部使用,不做讨论

4.poll:轮询

处理I/O事件的回调,适当时候,Node将在这里堵塞

5.cheak

setImmediate的回调在此执行

6.close callback

关闭的回调,close事件的监听回调

重点解释

1.poll阶段

主要任务 1.计算定时器应该阻塞的时间 2.执行该阶段队列的所有回调函数(I/O回调) 3.如果队列为空,也就是没有工作可做,这是如果有setImmediate设置的回调存在,就会直接结束该阶段,不会等待响应的I/O回调,如果没有setImmediate设置的回调队列,就会等待I/O操作 举个例子 比如有一个设置了100ms后执行的定时器,同时有一个I/O操作,交给内核,内核正在读取文件,当事件循环开始时,timers阶段显示时间没到100ms,跳过进入poll阶段,此时文件还没有读完,但因为poll阶段检测到也没到定时器的100ms,所以即使进行下一轮的事件循环还是会跳过timers阶段,所以决定等待文件读取的操作,就是堵塞在poll阶段,直到100ms,此时如果文件读取完成,就会把文件读取的回调执行完,再进入下一轮事件循环,执行timers的定时器

但有一个情况会有所不同,就是在cheak阶段的队列不为空,即有setImmediate设置的回调,此时的poll阶段就不会等待I/O操作,而是会直接清空该阶段的任务队列再进入cheak阶段,清空cheak阶段的任务队列。

2.setImmediate和setTimeout

用于不同,setImmediate意为在本轮I/O操作后马上执行回调,setTimeout则是一段时间后,尽可能快的执行回调

3.process.nextTick

独立维护一个队列,在每个阶段结束后,都会优先清空该队列

node中的事件循环和宏任务微任务

宏任务:

setTimeout,setInterval,setImmediate I/O回调 同步脚本

微任务:

process.nxetTick promise.then catch finall 注意:process.nextTick优先级大于promise

每处理一个宏任务都会处理掉所有微任务