node.js 实战——(概念以及Buffer 知识点学习)

概念

node.js是一个开源的、跨平台的javascript运行环境;它可以开发服务器应用,可以开发工具类应用(webpack、vite、Babel),也可以开发桌面端应用(vscode、Figma、Postman)
node electron vscode figma postman

注意:

1)node.js 中不能使用BOM(Brower Object Model)和 DOM (Document Object Model)的API

  1. node.js 中的顶级对象为global ,也可以用globalThis访问顶级对象

node的特点

异步I/O

I/O 是 Input/Output 的缩写,指的是程序与外部进行"交流"的过程,比如:

• 从硬盘读取文件

• 向数据库写数据

• 网络请求发送/接收数据

  • 同步I/O
    与异步I/O对应的是同步 I/O,那什么是同步I/O?同步I/O就是等结果才干下一步。
javascript 复制代码
const fs = require('fs')

let data =fs.readFileSync("./file/2.mov");
console.log(data); //需要等待将文件读取完毕之后,才打印出这个data
  • 异步I/O
    异步 I/O是先安排任务,等会儿回来拿结果
javascript 复制代码
const fs = require('fs')
fs.readFile("./fileSystem.txt", "utf8", (err, data) => {
    if (err) {
        console.log(err)
    }
    console.log(data)
})
console.log("read file");

事件与回调函数

回调函数

回调函数是一个"等你处理完再叫我"的函数

事件

系统某个"动作"发生了,比如"文件读完了"、"请求完成了",它就会触发一个事件。

单线程

Node保持了JavaScript在浏览器中单线程的特点。

  • 好处
    单线程的最大好处是不用像多线程编程那样处处在意状态的同步问题,这里没有死锁的存在,也没有线程上下文交换所带来的性能上的开销。
  • 缺点
    无法利用多核CPU

针对缺点的解决方案:

1、使用 cluster 模块(内置)

• 启动多个 Node 进程,每个进程一个 CPU 核心。

• 主进程负责分发请求,子进程负责处理。

javascript 复制代码
const cluster = require('cluster');
const os = require('os');

if (cluster.isMaster) {
  const cpus = os.cpus().length;
  for (let i = 0; i < cpus; i++) {
 cluster.fork(); // 启动多个子进程
  }
} else {
  // 各个子进程独立跑服务
  require('./app'); 
}

2、使用 PM2(生产级进程管理器)

• 简化多进程管理,支持 cluster 模式,崩溃自动重启。

powershell 复制代码
pm2 start app.js -i max  # 根据 CPU 数自动 fork

错误回英气整个应用推出,应用的健壮性值得考验

解决方案

1、错误处理策略:全面 try/catch + 全局监听

2、使用中间件捕获错误(在框架中如 Express/NestJS)

3、 使用 PM2 进行守护重启

大量计算占用CPU导致无法继续调用异步I/O

解决方案

1、Worker Threads(Node 10.5+ 支持)

• Node.js 原生的多线程模块,适合重计算任务。

javascript 复制代码
const { Worker } = require('worker_threads');
const worker = new Worker('./heavy-task.js'); // 把大计算扔到 worker 去干
worker.on('message', result => {
console.log('计算结果:', result);
});

2、把计算任务 offload 到别的服务(如 Python)

• 比如图像处理、机器学习等,用子进程或远程服务来做

3、使用 WebAssembly

• 某些计算密集任务可以用 Rust/C 编译成 WebAssembly,在 Node 里跑得飞快且不阻塞。

跨平台

Buffer缓冲区

buffer 类似于Array的对象,用于表示固定长度的字节序列,用于处理二进制数据

中文在utf-8中占三个字节

Buffer 是一个典型的javascript与c++结合的模块,他将性能相关部分用C++实现,将非性能相关的部分用javascript实现。
C++内建模块 node_buffer javascript核心模块 Buffer/ slowBuffer

特点

  1. Buffer 大小固定且无法调整
  2. Buffer性能较好,可以直接对计算机内存进行操作
  3. 每个元素的大小为1字节(8位)

创建Buffer

alloc

javascript 复制代码
let buf =Buffer.alloc(10);

allocUnsafe

javascript 复制代码
let buf = Buffer.allocUnsafe(10);

alloc 和 allocUnsafe 的区别:

  • alloc 在创建的时候,会将申请到的内存数据清零
  • allocUnsafe创建时,可能包含旧的内存数据,但正因为保留了之前的数据,速度比alloc快

from

将字符串/数组转换成buffer,每一个字符会先转换成unicode对应的,然后再转二进制

javascript 复制代码
let buf = Buffer.from("你好");

溢出

Buffer 每个元素的大小为1字节,也就是8位,能表示的最大值为 255 即11111111;那么当超过255之后,程序会如何处理呢。

这里并没有输出361,而是105 ,这是因为361 的二进制是 0001 0110 1001 ;当超过255之后,会舍弃高位,即舍去了高位的0001 ,就剩下0110 1001 (十进制:105)

Buffer与字符串的转换

可以直接借助toString方法将Buffer转位字符串

字符串转Buffer

主要是通过构造函数完成的

javascript 复制代码
new Buffer(str,[encoding])

Buffer 内存分配

Buffer对象的内存分配不是在V8的堆内存中,而是在node的C++层面实现内存的申请。

为了高效地使用申请来的内存,node采用的slab分配机制。slab简单来说,就是一块申请好的固定大小的内存区域。

应用场景

在文件I/O 和 网络I/O 中运用广泛

相关推荐
AI 编程助手GPT13 分钟前
用 Python 做一个世界杯赛前分析脚本:以巴西 vs 摩洛哥为例
开发语言·网络·人工智能·python·chatgpt
lihao lihao27 分钟前
Linux信号
开发语言·c++·算法
独泪了无痕32 分钟前
Vue3中防御XSS攻击的“特效药”-DOMPurify
前端·vue.js·安全
Java患者·38 分钟前
《Python 人脸识别入门实践:从人脸检测到人脸比对完整实现》
开发语言·python·opencv·目标检测·计算机视觉·目标跟踪·视觉检测
ceclar12339 分钟前
C# 的任务并行库(TPL)
开发语言·c#·.net
小小199243 分钟前
idea 配置less转化为css
前端·css·less
hhb_6181 小时前
Less嵌套避坑:优先级冲突实战解析
前端·css·less
快乐的哈士奇1 小时前
【Next.js实战①】Gmail API 按柜号检索邮件:OAuth 双 Cookie 与搜索 Fallback
开发语言·javascript·ecmascript
weixin_307779131 小时前
Python写入Shell文件使用Linux系统的换行符
linux·开发语言·python·自动化
云水一下1 小时前
Vue.js从零到精通系列(五):全局状态管理——Pinia 核心与实践
前端·javascript·vue.js