专栏首页Nodejs技术栈你觉得 Node.js 是单线程这个结论对吗?

你觉得 Node.js 是单线程这个结论对吗?

人的一生就是进行尝试,尝试的越多,生活就越美好。

——爱默生

前言

一提到 Node.js ,我想大家都会想到它的一个特点,单线程。但是 Node.js 在运行的时候依赖 V8 这个宿主环境,难道在宿主环境中也是单线程吗?请看正文解释你这个疑惑。

Node.js 单线程误区

开启 Node.js 服务 Demo

const http = require('http');
const server = http.createServer();server.listen(3000,()=>{    process.title='程序员成长指北测试进程';    console.log('进程id',process.pid)})

看这段代码,创建了 http 服务,开启了一个进程,都说了 Node.js 是单线程,所以 Node.js 启动后线程数应该为 1,但是事实并非如此,呜呜呜。

看活动监视器怎么开启7个线程呢?难道Javascript不是单线程不知道小伙伴们有没有这个疑问?

解释一下这个原因:

Node.js 中最核心的是 V8 引擎,在 Node.js 启动后,会创建 V8 的实例,这个实例是多线程的。

  • 主线程:编译、执行代码。
  • 编译/优化线程:在主线程执行的时候,可以优化代码。
  • 分析器线程:记录分析代码运行时间,为 Crankshaft 优化代码执行提供依据。
  • 垃圾回收的几个线程。

所以大家常说的 Node.js 是单线程的指的是 JavaScript 的执行是单线程的(开发者编写的代码运行在单线程环境中),但 Javascript 的宿主环境,无论是 Node 还是浏览器都是多线程的因为 libuv 中有线程池的概念存在的,libuv 会通过类似线程池的实现来模拟不同操作系统的异步调用,这对开发者来说是不可见的。

V8中什么时候会创建额外进程

某些异步 IO 会占用额外的线程

还是上面那个例子,我们在定时器执行的同时,去读一个文件:

const fs = require('fs')setInterval(() => {    console.log(new Date().getTime())}, 3000)
fs.readFile('./kaola.html', () => {})

线程数量变成了 11 个,这是因为在 Node 中有一些 IO 操作(DNS,FS)和一些 CPU 密集计算(Zlib,Crypto)会启用 Node 的线程池,而线程池默认大小为 4,因为线程数变成了 11。

手动更改线程池默认大小:

process.env.UV_THREADPOOL_SIZE = 64

修改线程池默认大小后,轻松把线程变成 71。

Libuv

Libuv 是一个跨平台的异步 IO 库,它结合了 UNIX 下的 libev 和 Windows 下的 IOCP 的特性,最早由 Node.js 的作者开发,专门为 Node.js 提供多平台下的异步IO支持。Libuv 本身是由 C++ 语言实现的,Node.js 中的非阻塞 IO 以及事件循环的底层机制都是由 libuv 实现的。

libuv架构图

在 Windows 环境下,libuv 直接使用Windows的 IOCP 来实现异步IO。在 非Windows 环境下,libuv使用多线程(线程池Thread Pool)来模拟异步IO,这里仅简要提一下 libuv 中有线程池的概念,之后的文章会介绍 libuv 如何实现进程间通信。

注意下面我要说的话:

Node的异步调用是由 libuv 来支持的,以上面的读取文件的例子,读文件实质的系统调用是由 libuv 来完成的,Node只是负责调用 libuv 的接口,等数据返回后再执行对应的回调方法。

总结

本篇文章仅对 Node.js 的单线程误区做了讲解,不过本篇文章只是 Node.js 高级进阶之进程与线程的 预热篇,接下来的文章会 对 Node.js 的进程与线程做一个详细讲解,包括原理分析,child_process 模块与 cluster模块,进程守护以及在真实项目中的 Node.js 多进程架构模型等内容。

本文分享自微信公众号 - Nodejs技术栈(NodejsDeveloper)

原文出处及转载信息见文内详细说明,如有侵权,请联系 yunjia_community@tencent.com 删除。

原始发表时间:2019-08-05

本文参与腾讯云自媒体分享计划,欢迎正在阅读的你也加入,一起分享。

我来说两句

0 条评论
登录 后参与评论

相关文章

  • linux内核级同步机制--futex

    在面试中关于多线程同步,你必须要思考的问题 一文中,我们知道glibc的pthread_cond_timedwait底层是用linux futex机制实现的。

    李红
  • 接口默认方法和 Lambda 表达式

    Java8 使开发者能够通过使用 default 关键字向接口添加非抽象方法实现,也称为虚拟扩展方法。

    happyJared
  • 【新手教程】手把手教你搭建腾讯云服务器

        暑假期间,愁着无聊但也不能荒废学业吧,毕竟以后想靠技术混口饭吃!为了实施自己的计划,特地挑了一个便宜的云服务器来用作自己的后台;这不是学生狗没钱嘛,所以...

    赤孺
  • LeetCode 138:复制带随机指针的链表 Copy List with Random Pointer

    给定一个链表,每个节点包含一个额外增加的随机指针,该指针可以指向链表中的任何节点或空节点。

    爱写bug
  • Java8新特性:默认方法详解

    Java8 中的默认方法是针对接口添加的新特性,它是指接口可以直接对方法进行实现,实现方式很简单,直接在方法定义处添加 default 关键字即可,如下所示。

    南风
  • CGLIB动态代理实现原理

    CGLIB(Code Generation Library)是一个开源项目!是一个强大的,高性能,高质量的Code生成类库,

    Java栈
  • SpringMVC入门看这一篇就够了!

    SpringMVC是Spring家族的一员,Spring是将现在开发中流行的组件进行组合而成的一个框架!它用在基于MVC的表现层开发,类似于struts2框架

    乔戈里
  • Java描述设计模式(05):原型模式

    知了一笑
  • 聊聊spring cloud的consulRetryInterceptor

    本文主要研究一下spring cloud的consulRetryInterceptor

    codecraft
  • java学习笔记(基础篇)—IO流

    流就是一个对象。所在的包java.io.*,Java I/O系统负责处理程序的输入和输出,java.io包它对各种常见的输入流和输出流进行了抽象。

    chlinlearn

扫码关注云+社区

领取腾讯云代金券