nodejs-第二章-第二节-nodejs事件循环(2-1)

当Node.js启动时会初始化event loop,每一个event loop 都会包含按以下六个节点循环,nodejs事件循环和浏览器事件循环彻底不同。

nodejs事件循环

  • 图中的每一个方框被称做事件循环的一个"阶段",这6个阶段为一轮事件循环;
阶段概览
  • timers(定时器): 此阶段执行那些有setTimeout() 和 setInterval()调度的回调函数;
  • I/O callbacks(I/O回调):此阶段会执行几乎全部的回调函数,除了close callbacks(关闭回调) 和 那些有timers与setImmediate()调度的回调;
  • idle(空转),prepare: 此阶段只在内部使用;
  • poll(轮询): 检索新的I/O事件;在恰当的时候Node会阻塞在这个节点;
  • check(检查): setImmediate() 设置的回调会在此阶段被调用;
  • close callbacks(关闭事件的回调):
    • 如:socket.on('close',...);此类的回调会在此阶段被调用; 在事件循环每次运行之间,Node.js会检查它是否等待异步I/O或定时器,若是没有的话就会自动关闭;

若是event loop 进入了poll阶段,且代码未设定timer,将会发生下面的状况:node

  • 若是poll queue不为空,event loop将同步的执行queue里的callback,知道queue为空,或执行的callback达到系统的上限;
  • 若是poll queue为空,将会发生下面的状况:
    • 若是代码已经被setImmediate()设定了callback,event loop 将结束poll阶段进入check阶段,并执行check阶段的queue(check阶段的queue是setImmediate设定的)
    • 若是代码没有设定setImmediate(callback),event loop将阻塞在该阶段等待callbacks加入poll queue,一旦达到就当即执行;

若是event loop进入了poll阶段,且代码设定了timer:浏览器

  • 若是poll queue进入空状态时(即poll阶段为空闲状态),event loop将阻塞在该阶段等待callbacks加入poll queue,一旦达到就当即执行;

setImmediate约定于setTimeout(cb,0)
  • path.resolve() 方法会把一个路径或路径片断的序列解析为一个绝对路径。异步

  • __dirname 老是指向当前文件夹的绝对路径socket

  • __filename 老是指向当前文件的绝对路径函数

  • note:oop

    • io: 浏览器线程去调用一些异步的回调
  • 执行代码1
var fs = require('fs');
var path = require('path');
function someAsyncOperation(callback) {
  // 花费2ms
  fs.readFileSync('./read.txt', callback);
}
var timeoutScheduled = Date.now();
var fileReadTime = 0;
setTimeout(function() {
  var delay = Date.now() - timeoutScheduled;
  console.log(`setTimeout ${delay} ms have passed since I was sheculed`);
  console.log('fileReaderTime', fileReadTime - timeoutScheduled);
}, 10);

someAsyncOperation(function() {
  fileReadtime = Date.now();
  while (Date.now() - fileReadTime < 20) {}
});
  • 执行过程:
    • setTimeout和readFile前后加入io
    • setTimeout执行进入io;(须要10ms)
    • readfile执行,也进入io;(须要2ms)
    • 2ms以后, redfile已经读取完毕,加入poll队列,此时poll为空,执行someAsyncOperation回调;
    • 因为此回调有while,这里阻塞20ms;执行完为22ms
    • 在10ms时,setTimeout不能执行,由于js是单线程,setTimeout一直被阻塞
    • 执行完以后(22ms之后),poll 进入空闲状态
    • event loop检查timer,setTimeout回调;
  • 执行结果:
    • readFile执行22ms
    • setTimeout执行22ms以后
  • 执行代码2
var fs = require('fs');
var path = require('path');
function someAsyncOperation(callback) {
  // 花费9ms
  fs.readFileSync('./read.txt', callback);
}
var timeoutScheduled = Date.now();
var fileReadTime = 0;
setTimeout(function() {
  var delay = Date.now() - timeoutScheduled;
  console.log(`setTimeout ${delay} ms have passed since I was sheculed`);
  console.log('fileReaderTime', fileReadTime - timeoutScheduled);
}, 5);

someAsyncOperation(function() {
  fileReadtime = Date.now();
  while (Date.now() - fileReadTime < 20) {}
});
  • 执行结果
    • setTimeout执行:5ms
    • readFile执行9 ~ 29ms
  • 执行代码3
  • 在nodejs中,setTimeout(fn,0) === setTimeout(fn,0)
  • 在浏览器中,setTimeout(fn,0) === setTimeout(fn,4)
setImmediate(() =>{
    console.log('setImmediate')
},0)

setTimeout(() =>{
    console.log('setTimeout')
},0)

// setTimeout 和 setImmediate的执行顺序不肯定
// 由于event loop的启动也是须要时间的,可能执行到poll阶段时已经超过了1ms,此时setTimeout会先执行
const fs = require('fs');
const path = require('path');
fs.readFile(path.resolve(__dirname, '/read.txt'), () => {
  setTimeout(() => {
    console.log('setTimeout');
  }, 0);
  setImmediate(() => {
    console.log('setImmediate');
  }, 0);
});
// 执行顺序是肯定的, setImmediate,setTimeout
  • 执行过程:
    • 执行fs,第一轮主线程没有timer和setImmediate
    • 假设readFile须要2ms,执行回调
    • 由poll进入check,setImmediate会先调用
    • 在第二轮的timer阶段,会执行setTimeout
相关文章
相关标签/搜索