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. 事件循环在适当的轮询阶段取出回调,在主线程上执行。

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

相关推荐
风止何安啊15 小时前
网课倍速痛点解决:一套前端代码实现自由控速播放器
前端·javascript·node.js
糖拌西瓜皮16 小时前
Node.js核心模块实战:文件、路径、HTTP与流处理
javascript·node.js
糖拌西瓜皮16 小时前
Node.js工程化实践:包管理、TypeScript配置与代码质量
typescript·node.js
糖拌西瓜皮16 小时前
NestJS入门指南:Java开发者的Spring Boot体验
javascript·node.js
糖拌西瓜皮16 小时前
Express框架快速上手:中间件、路由与错误处理
javascript·node.js
半个落月19 小时前
从 Tokenization 到 Embedding:用 Node.js 搞懂大模型为什么先“分词”再“向量化”
人工智能·node.js
叁两2 天前
前端转型AI Agent该如何学习?(前置篇)
前端·人工智能·node.js
糖拌西瓜皮2 天前
TypeScript 进阶:泛型、条件类型、类型守卫与装饰器
javascript·node.js
米丘5 天前
vite8 vite preview 命令做了什么?
node.js·vite
blanks20206 天前
生成 公钥私钥 笔记
node.js