node.js 的第一个基本观点是,I/O 操作是昂贵的:

目前的编程技术最大的浪费来自等待 I/O 操作的完成。有几种方法可以解决这些对性能的影响(来自Sam Rushing):
  同步:依次处理单个请求。
    优点:简单。
    缺点:任何一个请求都会阻塞其余请求。
  创建新进程:为每个请求创建一个进程处理
    优点:容易。
    缺点:扩展性不好,数百个连接意味着数百个进程。fork()是 Unix 程序员的锤子。因为它很有用,所有的问题都像是钉子。但这通常是多余的。
  线程:为每个请求创建一个线程处理。
    优点:容易;由于线程的开销通常都很小,相比于使用 fork 对内核更友好。
    缺点:你的机器可能没有线程,并且线程编程很容易变得复杂,也存在如何访问共享资源的问题。

第二个基本观点是,单线程连接非常消耗内存。

Apach 是多线程的:为每一个请求创建一个线程(或者进程,这取决于配置)。你可以看到增加当前连接数是如何消耗内存的,多个线程需要同时服务多个客户。Nginx 和 Node.js 不是多线程的,因为多线程和多进程会带来沉重的内存开销。它们是单线程的,但是基于事件的。通过单线程处理多个连接,解决数千个线程/进程的开销问题。

Node.js 为代码保持着单线程的运行环境

Node.js 确实是单线程运行的:你不能执行任何并发代码;例如“sleep”,这会使服务器停止。

当代码运行时,node.js不会响应客户端的其他请求,因为它只有一个线程在执行代码。或者你可以使用一些 CPU-密集型代码,例如,调整图片尺寸,这仍然会阻塞其他请求。

 然而,一切代码都能并行执行

并没有办法让代码在单线程中并行运行。除了所有的 I/O 操作和异步事件,以下代码并不会阻塞服务器:[codesyntax lang="javascript"]

在一个请求中执行以上代码,数据库在休眠时其他请求也能被很好的处理。

这样有什么好处?我们什么时候应该将同步改为异步/并行执行?

同步执行是好方法,因为这使代码编写变得简单(与多线程相比,并发问题导致了 WTFs)。

在 node.js 中,你不需要担心后台会发生什么:只需要使用回调执行 I/O 操作;这保证了你的代码不会被中断,同时 I/O 操作不会阻塞其他请求,每个请求也不会增加线程/进程的开销(例如,Apache 中的内存开销)。

异步 I/O 操作也是好方法,因为 I/O 操作相较于大多数代码的执行更昂贵,我们应该做其他的事情,而不是等待 I/O 操作

时间循环是“一个能够加工和处理外部事件并将它们转换为回调调用的实体”。因此 I/O 调用的关键在于 Node.js 能够从一个请求切换到另一个请求。在一个 I/O 调用中,代码会保存回调函数,并将控制权返回给 node.js 的运行时环境。当数据可用时回调函数将被调用。

当然,在后台中,有用于数据库访问和执行进程的线程和进程。然而,这并没有使代码暴露,因此你不需要为 I/O 操作担心,例如,数据库或者其他进程对于每一个请求都是异步的,这些线程的执行结果会通过事件循环返回给代码。与 Apache 模式相比,不需要为每个连接提供单个线程,因此需要更少的线程和线程开销;只有当真的需要并行运行时,即使管理权在 Node.js 也能够运行。

除了 I/O 操作的调用,Node.js 希望其他的所有请求都能迅速响应;例如:CPU-密集型工作应该被拆分到交互事件的进程中,或者像WebWorkers 那样抽象的使用。(显然地)这意味着在后台没有其他的线程并发运行交互事件。基本上,所有的监听事件对象(都是 EventEmitter 的实例)都支持异步交互事件,你能够以这种方式与阻塞代码交互,例如使用 files,sockets 或者子进程,这些在 Node.js 中都是 EventEmitters。[多核][8]也可以使用这种方法,请参见:node-http-proxy

内部实现

内部,node.js 依赖于 libev 实现事件循环,以 libeio 为辅助,使用混合线程实现异步 I/O 操作。要想学习更多,就需要查看 libev 的文档

如何在 Node.js 中使用异步?

Tim Caswell 在他出色的演讲中描述了这种模式:

  First-class 函数。例如,我们将函数作为参数传递,在需要的时候执行他们。
  Function 形式。也被称作匿名函数或者闭包函数,当 I/0 操作完成后执行。

原文:Understanding the node.js event loop

理解 node.js 的事件循环的更多相关文章

  1. 理解Node.js的事件轮询

    前言 总括 : 原文地址:理解Node.js的事件轮询 Node小应用:Node-sample 智者阅读群书,亦阅历人生 正文 Node.js的两个基本概念 Node.js的第一个基本概念就是I/O操 ...

  2. Node.js 的事件循环机制

    目录 微任务 事件循环机制 setImmediate.setTimeout/setInterval 和 process.nextTick 执行时机对比 实例分析 参考 1.微任务 在谈论Node的事件 ...

  3. Node.js:事件循环

    ylbtech-Node.js:事件循环 1.返回顶部 1. Node.js 事件循环 Node.js 是单进程单线程应用程序,但是通过事件和回调支持并发,所以性能非常高. Node.js 的每一个 ...

  4. 【node.js】事件循环、EventEmitter

    Node.js 是单进程单线程应用程序,但是通过事件和回调支持并发,所以性能非常高. Node.js 的每一个 API 都是异步的,并作为一个独立线程运行,使用异步函数调用,并处理并发. 事件驱动程序 ...

  5. [译] 所有你需要知道的关于完全理解 Node.js 事件循环及其度量

    原文地址:All you need to know to really understand the Node.js Event Loop and its Metrics 原文作者:Daniel Kh ...

  6. JS JavaScript事件循环机制

    区分进程和线程 进程是cpu资源分配的最小单位(系统会给它分配内存) 不同的进程之间是可以同学的,如管道.FIFO(命名管道).消息队列 一个进程里有单个或多个线程 浏览器是多进程的,因为系统给它的进 ...

  7. 方便大家学习的Node.js教程(一):理解Node.js

    理解Node.js 为了理解Node.js是如何工作的,首先你需要理解一些使得Javascript适用于服务器端开发的关键特性.Javascript是一门简单而又灵活的语言,这种灵活性让它能够经受住时 ...

  8. 深入理解Node.js中的垃圾回收和内存泄漏的捕获

    深入理解Node.js中的垃圾回收和内存泄漏的捕获 文章来自:http://wwsun.github.io/posts/understanding-nodejs-gc.html Jan 5, 2016 ...

  9. 深入理解QStateMachine与QEventLoop事件循环的联系与区别

    最近一直在倒腾事件循环的东西,通过查看Qt源码多少还是有点心得体会,在这里记录下和大家分享.总之,对于QStateMachine状态机本身来说,需要有QEventLoop::exec()的驱动才能支持 ...

随机推荐

  1. MySQL复制格式小结

    基于语句级的复制 binlog=statement   优点: (1)binlog文件较小. (2)日志是包含用户执行的原始SQL,方便统计和审计. (3)出现最早可binlog.兼容较好. (4)b ...

  2. [SVG] Combine Multiple SVGs into an SVG Sprite

    In this lesson, we’ll explore the process of combining all of your SVG icons into one SVG sprite, to ...

  3. TensorFlow 需注意的细节问题

    1. 数据类型 不带小数点的数默认为 int32,带小数点的数默认为 float32:

  4. 使用CreateThread函数和_beginThreadex函数的注意事项

    作者:朱金灿 来源:http://blog.csdn.net/clever101 使用CreateThread函数创建线程时,类或结构体的变量作为CreateThread函数传递给线程函数的参数需要避 ...

  5. APP和服务端-架构设计(一)

    架构因人而异,不同的架构师大多会有不同的看法:架构也因项目而异,不同的项目需求不同,相应的架构也会不同.然而,有些东西还是通用的,是所有架构师都需要考虑的,也是所有项目都会有的需求,比如API如何设计 ...

  6. 深入Qt 学习 -- 反射机制(比较简单清楚)

    相对于Java天生的这一特性, C++并不具备;但进入到Qt领域,这一切都变得简单自如了. 从Qt的元对象系统可知,除了提供信号/槽机制的特性之外,它还提供了以下特性: ■ QObject::meta ...

  7. robot framework的使用说明

    robot framework安装说明1.安装python2.7.15运行安装包python-2.7.15.amd64.msi 2.robot framework(1)解压最新的压缩包如robotfr ...

  8. zedboard之GPIO驱动器(离FPGA直到LINUX申请书)

    笔者:xiabodan   资源: http://blog.csdn.net/xiabodan/article/details/24308373 1 EDK 大家知道我们在EDK中建立GPIO然后倒出 ...

  9. C#中的String.Format介绍

    关键字:C# string.format作者:txw1958原文:http://www.cnblogs.com/txw1958/archive/2012/11/15/csharp-string_for ...

  10. Go语言的网络功能太强了,这么多项目。。。

    Centrifugo 是一个用 Golang 实现的基于 Websocket 或者 SockJS 的实时通信平台.https://www.oschina.net/p/centrifugalrpcx是一 ...