Node.js基础-进程、线程、线程池

Node.js基础-进程、线程、线程池

Node.js进程:程序的运行实例

在操作系统中,进程是程序的一次执行实例,它拥有独立的内存空间和至少一个执行线程。你每次运行node app.js,实际上就是启动一个Node.js进程;

当进程启动的时候,会发生什么?

  • 初始化程序: 加载Node.js运行时环境;
  • 执行顶层代码:从入口文件开始,自上而下执行不再回调函数内的"顶层"同步代码;
  • 加载模块:处理我们代码中的require或import,构建模块依赖树;
  • 注册事件回调:当遇到fs.readFile(),server.listen()等异步调用时,会将其回调注册到事件循环的相应阶段;
  • 启动事件循环:这是Node.js的心脏,只要还有未处理的事件或回调,它就会持续运转;

单线程:Event Loop的主战场

Node.js本身在JavaScript层面是单线程运行的,这个单线程指的是事件循环运行的主线程;

  • 这个主线程不停轮询:检查调用栈是否为空、是否有待执行的微任务、是否有到期的定时器等;
  • 它可以处理成千上万的并发连接:由于主线程不负责执行耗时任务,只负责任务的分发和结果的回调,所以这种I/O的切换就会变的非常高效;

单线程的优势

  • 不需要担心多线程编程中的锁、竞态、死锁等问题;
  • 内存开销小,上下文切换成本极低;

单线程的软肋

  • 如果主线程执行了CPU密集型任务,比如大循环、复杂加密,这样就是占满整个调用栈,导致事件循环卡死,其他所有的回调都无法及时执行;

线程池

为了解决某些无法由操作系统内核异步万恶很难过的耗时任务,Node.js借助libuv提供了线程池

默认配置

  • libuv的线程池默认包含了4个线程
  • 但是你可以通过环境变量设计线程数,但是最大不建议超过CPU核心数;

线程池负责的工作

  • 文件系统API:如fs.readFile、fswriteFile等。文件I/O在大多数操作系统中缺少真正的异步内核接口,所有由线程池模拟异步;
  • Cryptography:如 crypto.pbkdf2crypto.scrypt 等CPU密集的加密操作。
  • Compressionzlib 库的压缩/解压操作,计算量大,会在线程池中执行。
  • DNS查询dns.lookup 会使用底层的 getaddrinfo 调用,该调用在线程池中执行;而 dns.resolve 则使用 c-ares 库,是真正的异步网络调用,不走线程池。

工作流程

  1. 主线程执行到这些耗时操作时,将任务和回调提交给线程池;
  2. 线程池中的一个空闲线程会取走任务并在后台执行;
  3. 执行完毕后,把回调连同结果绑定到事件循环的某个I/O观察者上;
  4. 事件循环在适当的轮询阶段取出回调,在主线程上执行。

这种设计使得主线程永远保持轻量、响应迅速

相关推荐
m0_535817552 小时前
macOS下Claude Code从0到1配置教程(附API密钥获取+常见报错修复)
gpt·macos·node.js·api·claude·claudecode·88api
用户357085028815 小时前
我做了一个自动生成项目入门文档的 CLI 工具
node.js
云水一下8 小时前
模块系统与 npm——万物皆模块
前端·npm·node.js
wgc2k10 小时前
Node.js游戏服务器项目移植 4-MongoDB的移植
mongodb·游戏·node.js
meilindehuzi_a11 小时前
Node.js × 大模型:AIGC 工程化基础与异步流控总结
node.js·aigc
不好听6131 天前
Node.js 工程化开发流程 — 知识点总结
javascript·node.js
HjhIron1 天前
🚀 从零开始,用 Node.js 构建你的第一个 AIGC 项目
node.js·aigc
To_OC1 天前
我调用 DeepSeek API 连踩 3 个坑,终于把 Node AIGC 开发的核心知识点捋顺了
后端·node.js·aigc
sugar__salt1 天前
从零落地 Generative AI 接口调用:Node.js 工程化最佳实践
人工智能·node.js