Node.js事件循环(Event Loop)和线程池详解


Posted in Javascript onJanuary 28, 2015

Node的“事件循环”(Event Loop)是它能够处理大并发、高吞吐量的核心。这是最神奇的地方,据此Node.js基本上可以理解成“单线程”,同时还允许在后台处理任意的操作。这篇文章将阐明事件循环是如何工作的,你也可以感受到它的神奇。

事件驱动编程

理解事件循环,首先要理解事件驱动编程(Event Driven Programming)。它出现在1960年。如今,事件驱动编程在UI编程中大量使用。JavaScript的一个主要用途是与DOM交互,所以使用基于事件的API是很自然的。

简单地定义:事件驱动编程通过事件或状态的变化来进行应用程序的流程控制。一般通过事件监听实现,一旦事件被检测到(即状态改变)则调用相应的回调函数。听起来很熟悉?其实这就是Node.js事件循环的基本工作原理。

如果你熟悉客户端JavaScript的开发,想一想那些.on*()方法,如element.onclick(),他们用来与DOM元素相结合,传递用户交互。这个工作模式允许在单个实例上触发多个事件。Node.js通过EventEmitter(事件发生器)触发这种模式,如在服务器端的Socket和 “http”模块中。可以从一个单一实例触发一种或一种以上的状态改变。

另一种常见的模式是表达成功succeed和失败fail。现在一般有两种常见的实现方式。首先是将“Error异常”传入回调,一般作为第一个参数传递给回调函数。第二种即使用Promises设计模式,已经加入了ES6。注* Promise模式采用类似jQuery的函数链式书写方式,以避免深层次的回调函数嵌套,如:

$.getJSON('/getUser').done(successHandler).fail(failHandler)

“fs”(filesystem)模块大多采用往回调中传入异常的风格。在技术上触发某些调用,例如fs.readFile()附加事件,但该API只是为了提醒用户,用来表达操作成功或失败。选择这样的API是出于架构的考虑,而非技术的限制。

一个常见的误解是,事件发生器(event emitters)在触发事件时也是天生异步的,但这是不正确的。下面是一个简单的代码片段,以证明这一点。

function MyEmitter() {

  EventEmitter.call(this);

}

util.inherits(MyEmitter, EventEmitter);
MyEmitter.prototype.doStuff = function doStuff() {

  console.log('before')

  emitter.emit('fire')

  console.log('after')}

};
var me = new MyEmitter();

me.on('fire', function() {

  console.log('emit fired');

});
me.doStuff();

// 输出:

// before

// emit fired

// after
注* 如果 emitter.emit 是异步的,则输出应该为

// before

// after

// emit fired

EventEmitter经常表现地很异步,因为它经常用于通知需要异步完成的操作,但EventEmitter API本身是完全同步的。监听函数内部可以按异步执行,但请注意,所有的监听函数将按被添加的顺序同步执行。

机制概述和线程池

Node本身依赖多个库。其中之一是libuv,神奇的处理异步事件队列和执行的库。

Node利用尽可能多的利用操作系统内核实现现有的功能。像生成响应请求(request),转发连接(connections)并委托给系统处理。例如,传入的连接通过操作系统进行队列管理,直到它们可以由Node处理。

您可能听说过,Node有一个线程池,你可能会疑惑:“如果Node会按次序处理任务,为什么还需要一个线程池?”这是因为在内核中,不是所有任务都是按异步执行的。在这种情况下,Node.JS必须能在操作时将线程锁定一段时间,以便它可以继续执行事件循环而不会被阻塞。

下面是一个简单的示例图,来表示他内部的运行机制:

            ┌───────────────────────┐
?──►│         timers                                           │
 │         └───────────┬───────────┘
 │         ┌───────────┴───────────┐
 │         │   pending callbacks                             │
 │         └───────────┬───────────┘          ┌──────────────┐
 │         ┌───────────┴───────────┐          │  incoming:                    │
 │          │          poll                                               │◄──┤ connections,                │
 │         └───────────┬───────────┘          │  data, etc.                     │
 │         ┌───────────┴───────────┐          └──────────────┘
?───┤      setImmediate                                  │
             └───────────────────────┘

关于事件循环的内部运行机制,有一些理解困难的地方:

所有回调都会经由process.nextTick(),在事件循环(例如,定时器)一个阶段的结束并转换到下一阶段之前预设定。这就会避免潜在的递归调用process.nextTick(),而造成的无限循环。
“Pending callbacks(待回调)”,是回调队列中不会被任何其他事件循环周期处理(例如,传递给fs.write)的回调。

Event Emitter 和 Event Loop

通过创建EventEmitter,可简化与事件循环的交互。它是一个通用的封装,可以让你更容易地创建基于事件的API。关于这两者如何互动往往让开发者感到混乱。

下面的例子表明,忘记了事件是同步触发的,可能导致事件被错过。

// v0.10以后,不再需要require('events').EventEmitter 

var EventEmitter = require('events');

var util = require('util');
function MyThing() {

  EventEmitter.call(this);
  doFirstThing();

  this.emit('thing1');

}

util.inherits(MyThing, EventEmitter);
var mt = new MyThing();
mt.on('thing1', function onThing1() {

  // 抱歉,这个事件永远不会发生

});

上面的'thing1'事件,永远不会被MyThing()捕获,因为MyThing()必须在实例化后才能侦听事件。下面的是一个简单的解决方法,不必添加任何额外的闭包:
var EventEmitter = require('events');

var util = require('util');
function MyThing() {

  EventEmitter.call(this);
  doFirstThing();

  setImmediate(emitThing1, this);

}

util.inherits(MyThing, EventEmitter);
function emitThing1(self) {

  self.emit('thing1');

}
var mt = new MyThing();
mt.on('thing1', function onThing1() {

  // 执行了

});

下面的方案也可以工作,不过要损失一些性能:

function MyThing() {

  EventEmitter.call(this);
  doFirstThing();

  // 使用 Function#bind() 会损失性能

  setImmediate(this.emit.bind(this, 'thing1'));

}

util.inherits(MyThing, EventEmitter);

另一个问题是触发Error(异常)。找出您应用程序中的问题已经很难了,但没了调用堆栈(注* e.stack),则几乎不可能调试。当Error被远端的异步请求调用堆栈将丢失。有两个可行的解决方案:同步触发或确保Error跟其他重要信息一起传入。下面的例子演示了这两种解决方案:
MyThing.prototype.foo = function foo() {

  // 这个 error 会被异步触发

  var er = doFirstThing();

  if (er) {

    // 在触发时,需要创建一个新的保留现场调用堆栈信息的error

    setImmediate(emitError, this, new Error('Bad stuff'));

    return;

  }
  // 触发error,马上处理(同步)

  var er = doSecondThing();

  if (er) {

    this.emit('error', 'More bad stuff');

    return;

  }

}

审时度势。当error被触发时,是有可能被立即处理的。或者,它可能是一些琐碎的,可以很容易处理,或在以后再处理的异常。此外通过一个构造函数,传递Error也不是一个好主意,因为构造出来的对象实例很有可能是不完整的。刚才直接抛出Error的情况是个例外。

结束语

这篇文章比较浅显地探讨了有关事件循环的内部运作机制和技术细节。都是经过深思熟虑的。另一篇文章会讨论事件循环与系统内核的交互,并展现NodeJS异步运行的魔力。

Javascript 相关文章推荐
解析Jquery的LigerUI如何实现文件上传
Jul 09 Javascript
使用js完成节点的增删改复制等的操作
Jan 02 Javascript
js检测浏览器版本、核心、是否移动端示例
Apr 24 Javascript
JavaScript实现动态创建CSS样式规则方案
Sep 06 Javascript
javascript制作sql转换为stringBuffer的小工具
Apr 03 Javascript
浅谈jquery中delegate()与live()
Jun 22 Javascript
结合代码图文讲解JavaScript中的作用域与作用域链
Jul 05 Javascript
微信小程序 自己制作小组件实例详解
Dec 22 Javascript
JS实现中文汉字按拼音排序的方法
Oct 09 Javascript
如何将HTML字符转换为DOM节点并动态添加到文档中详解
Aug 19 Javascript
Vue从TodoList中学父子组件通信
Feb 05 Javascript
CocosCreator如何实现划过的位置显示纹理
Apr 14 Javascript
使用Sticker.js实现贴纸效果
Jan 28 #Javascript
javascript实现瀑布流自适应遇到的问题及解决方案
Jan 28 #Javascript
7个让JavaScript变得更好的注意事项
Jan 28 #Javascript
简单谈谈javascript代码复用模式
Jan 28 #Javascript
JS动态添加Table的TR,TD实现方法
Jan 28 #Javascript
扒一扒JavaScript 预解释
Jan 28 #Javascript
javascript弹出页面回传值的方法
Jan 28 #Javascript
You might like
实现php加速的eAccelerator dll支持文件打包下载
2007/09/30 PHP
基于MySQL到MongoDB简易对照表的详解
2013/06/03 PHP
php 获取今日、昨日、上周、本月的起始时间戳和结束时间戳的方法
2013/09/28 PHP
php设计模式之命令模式使用示例
2014/03/02 PHP
在Thinkphp中使用ajax实现无刷新分页的方法
2016/10/25 PHP
php版阿里云OSS图片上传类详解
2016/12/01 PHP
利用php生成验证码
2017/02/23 PHP
php下的原生ajax请求用法实例分析
2020/02/28 PHP
firefox下对ajax的onreadystatechange的支持情况分析
2009/12/14 Javascript
node.js中的fs.symlink方法使用说明
2014/12/15 Javascript
通过JS判断联网类型和连接状态的实现代码
2015/04/01 Javascript
JavaScript实现瀑布流图片效果
2017/06/30 Javascript
关于vue.js发布后路径引用的问题解决
2017/08/15 Javascript
vue3.0 CLI - 2.2 - 组件 home.vue 的初步改造
2018/09/14 Javascript
解决Layui 表格自适应高度的问题
2019/11/15 Javascript
搭建vscode+vue环境的详细教程
2020/08/31 Javascript
Python模块包中__init__.py文件功能分析
2016/06/14 Python
Python三级目录展示的实现方法
2016/09/28 Python
Python 高级专用类方法的实例详解
2017/09/11 Python
python微信跳一跳系列之棋子定位颜色识别
2018/02/26 Python
在Pycharm中修改文件默认打开方式的方法
2019/01/17 Python
python turtle 绘制太极图的实例
2019/12/18 Python
python3.7通过thrift操作hbase的示例代码
2020/01/14 Python
python如何保存文本文件
2020/06/07 Python
一套比较完整的软件测试人员面试题
2012/05/13 面试题
关于抽烟的检讨书
2014/02/25 职场文书
《水乡歌》教学反思
2014/04/24 职场文书
优秀毕业生找工作自荐信
2014/06/23 职场文书
员工教育培训协议书
2014/09/27 职场文书
中层领导干部群众路线对照检查材料思想汇报
2014/10/02 职场文书
2015年学校安全管理工作总结
2015/05/11 职场文书
师德师风培训感言
2015/08/03 职场文书
教你修复 Win11应用商店加载空白问题
2021/12/06 数码科技
python中mongodb包操作数据库
2022/04/19 Python
vue elementUI批量上传文件
2022/04/26 Vue.js
Python如何快速找到多个字典中的公共键(key)
2022/04/29 Python