如何理解 Node.js 不是完全的单线程的程序(浅析)

为什么说 node.js 不是完全的单线程?如何理解?下面本篇文章就来带大家探讨一下,希望对大家有所帮助!

如何理解 Node.js 不是完全的单线程的程序(浅析)

相信大家都知道 node 是一个单线程程序,使用了 Event Loop 可以做到多并发。可惜这是不完全正确的。

那么为什么说 Node.js 不是完全的单线程的程序呢?

Node.js 是单线程的程序*

所有我们自己写的 Javsacript,V8, event loop都跑在同一个线程里面,也就是 main thrad。

哎嗨,这不正说明 node 是单线程的吗?

但是也许你不知道 node 有很多模块背后都是 C++ code。

虽然 node 没有给使用者暴露控制 thread 的权限,但是 C++ 是可以使用多线程的。

那么什么时候 node 会使用多线程呢?

如果一个 node 方法,背后调用C++的同步方法,那么都是跑在 main thread 里面的。

如果一个 node 方法,背后调用C++的异步方法,有时候不是跑在 main thread 里面的。

Talk is cheap, show me the code.

同步方法,跑在 main thread 里面

这里 crypto 相关模块,很多是 C++ 写的。下面一段程序是计算hash的函数,一般用来存储密码。

import { pbkdf2Sync } from "crypto";const startTime = Date.now();let index = 0;for (index = 0; index < 3; index++) {    pbkdf2Sync("secret", "salt", 100000, 64, "sha512");    const endTime = Date.now();    console.log(`${index} time, ${endTime - startTime}`);}const endTime = Date.now();console.log(`in the end`);

输出的时间,

0 time, 44 1 time, 902 time, 134in the end

可以看到每次大概都是花费~45ms,代码 main thread 上顺序执行。

注意最后的输出是谁?注意这里一次 hash 在我的 cpu 需要~45ms。

异步 pbkdf2 方法,不跑在 main thread 里面

import { cpus } from "os";import { pbkdf2 } from "crypto";console.log(cpus().length);let startTime = console.time("time-main-end");for (let index = 0; index  {        if (err) throw err;        console.timeEnd(`time-${index}`);    });}console.timeEnd("time-main-end");

输出的时间,

time-main-end: 0.31mstime-2: 45.646mstime-0: 46.055mstime-3: 46.846mstime-1: 47.159ms

这里看到,main thread 早早结束,然而每次计算的时间都是45ms,要知道一个 cpu 计算 hash 的时间是45ms,这里 node 绝对使用了多个线程进行hash计算。

如果我这里把调用次数改成10次,那么时间如下,可以看到随着CPU核数的用完,时间也在增加。再一次证明node 绝对使用了多个线程进行hash计算。

time-main-end: 0.451mstime-1: 44.977mstime-2: 46.069mstime-3: 50.033mstime-0: 51.381mstime-5: 96.429ms // 注意这里,从第五次时间开始增加了time-7: 101.61mstime-4: 113.535mstime-6: 121.429mstime-9: 151.035mstime-8: 152.585ms

虽然这里证明了,node绝对启用了多线程。但是有一点点小小的问题?我的电脑的CPU是AMD R5-5600U,有6个核心12线程啊。但是为什么时间是从第五次开始增加的呢,node没有完全利用我的CPU啊?

原因是什么呢?

Node 使用了预定义的线程池,这个线程池的大小默认是4.

export UV_THREADPOOL_SIZE=6

豆包AI编程 豆包AI编程

豆包推出的AI编程助手

豆包AI编程 483 查看详情 豆包AI编程

让我们在看一个例子,

HTTP request

import { request } from "https";const options = {  hostname: "www.baidu.com",  port: 443,  path: "/img/PC_7ac6a6d319ba4ae29b38e5e4280e9122.png",  method: "GET",};let startTime = console.time(`main`);for (let index = 0; index  {    console.log(`statusCode: ${res.statusCode}`);    console.timeEnd(`time-${index}`);    res.on("data", (d) => {      // process.stdout.write(d);    });  });  req.on("error", (error) => {    console.error(error);  });  req.end();}console.timeEnd("main");
main: 13.927mstime-2: 83.247mstime-4: 89.641mstime-3: 91.497mstime-12: 91.661mstime-5: 94.677ms.....time-8: 134.026mstime-1: 143.906mstime-13: 140.914mstime-10: 144.088ms

这里主程序也早早结束了,这里我启动 http request 去下载15次图片,他们花费的时间并没有成倍增加,似乎不受限于线程池/cpu的影响。

为什么啊??Node 到底有没有在使用线程池啊?

如果 Node 背后的 C++ 的异步方法,首先会尝试是否有内核异步支持,比如这里网络请是使用 epoll (Linux),如果内核没有提供异步方式,Node才会使用自己的线程池。。

所以 http 请求虽然是异步,不过是由内核实现的,等到内核完成后,会通知C++, C++会通知给 main thread 处理callback。

那么 Node 哪些异步方法会使用线程池呢?哪些不会呢?

原生 Kernal Async

TCP/UDP server clientUnix Domain Sockets (IPC)pipesdns.resolveXXXtty input(stdin etc)Unix signalsChild process

Thread pool

fs.*dns.lookuppipe (edge case)

这也是大部分 Node 优化的切入点。

但是这些怎么和最重要的 Event Loop 结合起来呢?

Event Loop

相信大家都对 Event loop 非常熟悉了。Event loop 好比一个分发员,

如果是遇到普通 javascript 程序或者是 callback,交给 V8 处理。

如果遇到同步方法后背是 C++ 写的,交给C++,跑在 main thread。

如果遇到异步方法后背是 C++ 写的,如果有内核异步支持,从main thread 交给内核处理。

如果是异步方法后背是 C++ 写的,如果没有内核异步支持,从 main thread 交给 thread pool。

thread pool 和内核有结果都会把结果返回 event loop,如果注册的有 javascript callback,就交给V8进行处理。

然后如此循环,直到没有东西可以处理。

所以 Node 不完全是单线程程序。

更多node相关知识,请访问:nodejs 教程!

以上就是如何理解 Node.js 不是完全的单线程的程序(浅析)的详细内容,更多请关注创想鸟其它相关文章!

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。
如发现本站有涉嫌抄袭侵权/违法违规的内容, 请发送邮件至 chuangxiangniao@163.com 举报,一经查实,本站将立刻删除。
发布者:程序猿,转转请注明出处:https://www.chuangxiangniao.com/p/553278.html

(0)
打赏 微信扫一扫 微信扫一扫 支付宝扫一扫 支付宝扫一扫
上一篇 2025年11月9日 21:06:22
下一篇 2025年11月9日 21:09:32

相关推荐

  • 什么是智能合约(Smart Contract)?自动化协议的实现!

    2025主要加密货币交易平台 欧易OKX: Binance币安: 火币Huobi: Gateio芝麻开门: 智能合约,一个在当今数字世界中越来越被提及的词汇,它不仅仅是区块链技术的一个重要组成部分,更是一种能够颠覆传统商业模式的强大工具。想象一下,一份无需中介、自动执行的协议,一旦条件满足,结果便会…

    好文分享 2025年12月9日
    000
  • 什么是私链?企业级区块链的应用与特点!

    在数字经济的浪潮中,区块链技术以其去中心化、不可篡改和透明化的特性,正逐步改变着传统行业的运作模式。当人们谈论区块链时,往往首先想到的是比特币和以太坊等公有链,它们面向所有人开放,任何人都可以在上面进行交易和开发。然而,在企业级应用中,公有链的开放性有时反而成为了限制因素。这时,一种被称为“私链”的…

    好文分享 2025年12月9日
    000
  • 什么是联盟链?多方协作的区块链解决方案!

    什么是联盟链? 联盟链是一种介于公有链和私有链之间的区块链形式。它不像公有链那样完全开放,任何人都可自由参与;也不像私有链那样由单一实体完全控制。在联盟链中,参与的节点通常是经过授权和认证的机构。这些机构共同维护账本,并对交易进行验证。这种模式旨在平衡去中心化、效率、隐私和安全性,特别适用于需要多个…

    好文分享 2025年12月9日
    000
  • 交易所的API接口:自动化交易的利器

    2025主流交易所推荐: 欧易OKX: Binance币安: 火币Huobi: Gateio芝麻开门: 在高速发展的加密货币市场中,仅仅依靠手动操作已经难以满足专业交易者和机构的需求。这时,交易所提供的API接口便成为了连接交易策略与市场执行的桥梁,它允许用户通过编程方式与交易所进行交互,实现交易的…

    好文分享 2025年12月9日
    000
  • 以太坊在企业级应用中的潜力

    在数字经济浪潮席卷全球的当下,区块链技术作为底层基础设施,正以前所未有的速度改变着各个行业。其中,以太坊作为最成熟、最活跃的公链生态之一,其在企业级应用中的潜力吸引了无数目光。它不仅仅是一种加密货币,更是一个可编程的区块链平台,为开发者提供了构建去中心化应用(dapp)的强大工具。那么,以太坊究竟如…

    好文分享 2025年12月9日
    000
  • 数字黄金与智能合约:加密货币双雄

    在数字时代浪潮的拍打下,一种全新的资产形式——加密货币——正以前所未有的速度重塑着全球金融格局。它不仅仅是技术极客手中的玩物,更成为了各国央行、金融机构乃至普通投资者争相研究和布局的焦点。在众多加密货币中,“数字黄金”比特币与承载着无限创新潜力的智能合约平台以太坊,无疑是加密世界的两颗璀璨明星。它们…

    好文分享 2025年12月9日
    000
  • Fleek(FLK)币是什么?值得投资吗?Fleek工作原理、代币经济学及未来展望

    fleek 是一个由%ignore_a_1%驱动的社交平台,创作者和粉丝可以通过内置创作者代币和自动奖励来创作、混音和变现内容。该平台由 harrison hines 和 janison sivarajah 于 2019 年创立,总部位于纽约,已从 polychain capital、coinbas…

    2025年12月9日
    000
  • 分布式系统下的JavaScript消息队列实现

    答案:在Node.js中通过集成RabbitMQ或Kafka实现分布式系统消息通信。使用amqplib连接RabbitMQ,创建通道并声明交换机与队列,通过publish发送、consume接收消息,保障可靠性与解耦;或采用kafkajs连接Kafka集群,生产者向topic发消息,消费者订阅处理,…

    2025年12月6日 web前端
    000
  • JavaScript持续集成与部署

    持续集成与部署(CI/CD)通过自动化测试、构建和部署提升JavaScript项目交付效率。1. CI指频繁合并代码并自动运行测试以快速发现错误;2. CD在CI通过后自动将应用部署至生产环境;3. 常用工具包括GitHub Actions、GitLab CI/CD、CircleCI和Jenkins…

    2025年12月6日 web前端
    000
  • VSCode扩展包管理依赖解析

    VSCode扩展依赖通过package.json中的extensionDependencies声明,安装时自动解析并提示用户安装所需扩展,确保按顺序激活且禁止循环依赖,依赖间通过contributes.api共享功能,使用vsce打包时需手动处理生产依赖和性能优化,最终实现扩展间的协同运行与API调…

    2025年12月6日 开发工具
    000
  • Cloudinary 上传后临时文件未删除的解决方案与 React 错误排查

    本文旨在解决在使用 Cloudinary 进行文件上传后,临时文件未自动删除的问题,并提供针对 React UI 崩溃 “Objects are not valid as a React child” 错误的排查与修复方案。文章将深入探讨如何在文件上传完成后安全地删除临时文件…

    2025年12月6日 web前端
    000
  • VS Code扩展生态剖析:API设计与商店发布全流程指南

    VS Code扩展成功源于其插件化架构与丰富API。通过Activation Events、Contribution Points和Extension Host实现高效稳定的功能扩展,结合vscode.commands、languages、window、workspace等核心API提供完整开发支持…

    2025年12月6日 开发工具
    000
  • 如何配置VSCode以支持对容器内应用程序的远程调试?

    答案是使用VSCode Remote – Containers扩展结合Docker实现远程调试。首先安装Docker、VSCode及Remote – Containers扩展,然后在项目根目录创建.devcontainer文件夹并配置devcontainer.json,指定基…

    2025年12月6日 开发工具
    000
  • VS Code开发工坊:前端全栈开发环境搭建实战

    答案:通过安装ESLint、Prettier、Live Server、REST Client等核心插件,配置Node.js+Express后端环境并解决CORS实现前后端联调,利用launch.json设置断点调试,可构建高效VS Code全栈开发 workflow。 想用 VS Code 打通前端…

    2025年12月6日 开发工具
    000
  • 研究VSCode代码复杂度评估算法与重构建议系统

    VSCode通过集成ESLint、SonarLint等插件实现代码复杂度分析与重构建议,依赖LSP协议获取语义信息,支持圈复杂度、函数长度、嵌套层级等指标检测,并提供提取变量、重命名、语法优化等重构功能,结合自定义规则与AST分析可扩展高级功能,形成灵活的代码质量保障体系。 Visual Studi…

    2025年12月6日 开发工具
    000
  • JavaScript符号计算与代数系统

    符号计算指对数学表达式进行符号化操作,如化简、求导、解方程。JavaScript可通过math.js等库实现:支持表达式解析、简化(如2x+x→3x)、求导(如x²→2x),其核心是将表达式表示为抽象语法树(AST)。也可手动构建基础系统,用类模拟符号、加法、乘法等结构,适用于教育工具或轻量级交互场…

    2025年12月6日 web前端
    000
  • 在 JavaScript 项目中运行 TypeScript 子进程的实用指南

    本文详细介绍了在 javascript(如 electron)应用中以子进程方式运行 typescript 项目(如 express 服务器)时遇到的 `err_unknown_file_extension` 错误,并提供了通过 `node` 命令结合 `ts-node/esm` 加载器和 `exp…

    2025年12月6日 web前端
    000
  • 深入解析Google V8引擎:JavaScript代码执行的幕后机制

    google v8引擎作为高性能javascript运行时,其代码执行机制远超简单的抽象语法树(ast)解释器。v8通过解析、生成字节码并利用即时(jit)编译器将热点代码优化为高效机器码,实现了javascript的快速启动与极致性能。本文将详细探讨v8的编译与执行流程,并与基于ast的解释器进行…

    2025年12月6日 web前端
    000
  • 深入理解Google V8引擎:JavaScript代码执行机制解析

    本文深入探讨Google V8引擎如何执行JavaScript代码,对比了大学课程中常见的抽象语法树(AST)解释器模型与V8引擎先进的即时编译(JIT)技术。文章详细阐述了从源代码解析到机器码生成的各个阶段,包括词法分析、语法分析、字节码生成及优化编译,揭示了高性能JavaScript运行时的复杂…

    2025年12月6日 web前端
    000
  • JavaScript:判断对象数组中是否存在具有特定键值对的对象

    本文探讨了在javascript中如何高效地检查一个对象数组是否包含具有特定键值对的对象,并返回布尔值。我们将介绍两种主要方法:传统的循环遍历和现代的`array.prototype.some()`方法,并分析它们的优缺点及适用场景,帮助开发者根据具体需求选择最合适的实现方式。 在JavaScrip…

    2025年12月6日 web前端
    000

发表回复

登录后才能评论
关注微信