NodeJS模块Buffer原理及使用方法解析


Posted in NodeJs onNovember 11, 2020

Buffer 作为 nodejs 中重要的概念和功能,为开发者提供了操作二进制的能力。本文记录了几个问题,来加深对 Buffer 的理解和使用:

  • 认识缓冲器
  • 如何申请堆外内存
  • 如何计算字节长度
  • 如何计算字节长度
  • 如何转换字符编码
  • 理解共享内存与拷贝内存

认识 Buffer(缓冲器)

Buffer 是 nodejs 核心 API,它提供我们处理二进制数据流的功能。Buffer 的使用和 ES2017 的 Uint8Array 非常相似,但由于 node 的特性,专门提供了更深入的 api。

Uint8Array 的字面意思就是:8 位无符号整型数组。一个字节是 8bit,而字节的表示也是由两个 16 进制(4bit)的数字组成的。

const buf = Buffer.alloc(1);
console.log(buf); // output: <Buffer 00>

如何申请堆外内存

Buffer 可以跳出 nodejs 对堆内内存大小的限制。nodejs12 提供了 4 种 api 来申请堆外内存:

  • Buffer.from()
  • Buffer.alloc(size[, fill[, encoding]])
  • Buffer.allocUnsafe(size)
  • Buffer.allocUnsafeSlow(size)

Buffer.alloc vs Buffer.allocUnsafe

在申请内存时,可能这片内存之前存储过其他数据。如果不清除原数据,那么会有数据泄漏的安全风险;如果清除原数据,速度上会慢一些。具体用哪种方式,根据实际情况定。

  • Buffer.alloc:申请指定大小的内存,并且清除原数据,默认填充 0
  • Buffer.allocUnsafe:申请指定大小内存,但不清除原数据,速度更快

根据提供的 api,可以手动实现一个alloc:

function pollifyAlloc(size, fill = 0, encoding = "utf8") {
  const buf = Buffer.allocUnsafe(size);
  buf.fill(fill, 0, size, encoding);
  return buf;
}

Buffer.allocUnsafe vs Buffer.allocUnsafeSlow

从命名上可以直接看出效果,Buffer.allocUnsafeSlow更慢。因为当使用 Buffer.allocUnsafe 创建新的 Buffer 实例时,如果要分配的内存小于 4KB,则会从一个预分配的 Buffer 切割出来。 这可以避免垃圾回收机制因创建太多独立的 Buffer 而过度使用。

这种方式通过消除跟踪和清理的需要来改进性能和内存使用。

如何计算字节长度

利用 Buffer,可以获得数据的真实所占字节。例如一个汉字,它的字符长度是 1。但由于是 utf8 编码的汉字,所以占用 3 个字节。

直接利用Buffer.byteLength()可以获得字符串指定编码的字节长度:

const str = "本文原文地址: xxoo521.com";

console.log(Buffer.byteLength(str, "utf8")); // output: 31
console.log(str.length); // output: 19

也可以直接访问 Buffer 实例的 length 属性(不推荐):

console.log(Buffer.from(str, "utf8").length); // output: 31

如何转换字符编码

Nodejs 当前支持的编码格式有:ascii、utf8、utf16le、ucs2、base64、latin1、binary、hex。其他编码需要借助三方库来完成。

下面,是用Buffer.from()和buf.toString()来封装的 nodejs 平台的编码转换函数:

function trans(str, from = "utf8", to = "utf8") {
  const buf = Buffer.from(str, from);
  return buf.toString(to);
}

// output: 5Y6f5paH5Zyw5Z2AOiB4eG9vNTIxLmNvbQ==
console.log(trans("原文地址: xxoo521.com", "utf8", "base64"));

共享内存与拷贝内存

在生成 Buffer 实例,操作二进制数据的时候,千万要注意接口是基于共享内存,还是基于拷贝底层内存。

例如对于生成 Buffer 实例的from(),不同类型的参数,nodejs 底层的行为是不同的。

为了更形象地解释,请看下面两段代码。

代码 1:

const buf1 = Buffer.from("buffer");
const buf2 = Buffer.from(buf1); // 拷贝参数中buffer的数据到新的实例
buf1[0]++;

console.log(buf1.toString()); // output: cuffer
console.log(buf2.toString()); // output: buffer

代码 2:

const arr = new Uint8Array(1);
arr[0] = 97;

const buf1 = Buffer.from(arr.buffer);
console.log(buf1.toString()); // output: a

arr[0] = 98;
console.log(buf1.toString()); // output: b

在第二段代码中,传入Buffer.from的参数类型是arrayBuffer。因此Buffer.from仅仅是创建视图,而不是拷贝底层内存。buf1 和 arr 的内存是共享的。

在操作 Buffer 的过程中,需要特别注意共享和拷贝的区别,发生错误比较难排查。

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持三水点靠木。

NodeJs 相关文章推荐
nodejs中的fiber(纤程)库详解
Mar 24 NodeJs
PHP和NodeJs开发的应用如何共用Session
Apr 16 NodeJs
基于nodejs+express(4.x+)实现文件上传功能
Nov 23 NodeJs
Nodejs如何复制文件
Mar 09 NodeJs
Nodejs进阶:基于express+multer的文件上传实例
Nov 21 NodeJs
用nodejs搭建websocket服务器
Jan 23 NodeJs
NodeJS基础API搭建服务器详细过程记录
Apr 01 NodeJs
详解nodejs微信公众号开发——5.素材管理接口
Apr 11 NodeJs
NodeJS、NPM安装配置步骤(windows版本) 以及环境变量详解
May 13 NodeJs
详解HTTPS 的原理和 NodeJS 的实现
Jul 04 NodeJs
使用koa-log4管理nodeJs日志笔记的使用方法
Nov 30 NodeJs
NodeJS实现一个聊天室功能
Nov 25 NodeJs
nodejs中内置模块fs,path常见的用法说明
Nov 07 #NodeJs
Nodejs + sequelize 实现增删改查操作
Nov 07 #NodeJs
nodejs+koa2 实现模仿springMVC框架
Oct 21 #NodeJs
nodejs使用Sequelize框架操作数据库的实现
Oct 21 #NodeJs
用Nodejs实现在终端中炒股的实现
Oct 18 #NodeJs
Nodejs在局域网配置https访问的实现方法
Oct 17 #NodeJs
NodeJS开发人员常见五个错误理解
Oct 14 #NodeJs
You might like
php miniBB中文乱码问题解决方法
2008/11/25 PHP
php 3行代码的分页算法(求起始页和结束页)
2009/10/21 PHP
TP5框架使用QueryList采集框架爬小说操作示例
2020/03/26 PHP
javascript 异常处理使用总结
2009/06/21 Javascript
js 复制或插入Html的实现方法小结
2010/05/19 Javascript
从零开始学习jQuery (十一) 实战表单验证与自动完成提示插件
2011/02/23 Javascript
js将控件隐藏的方法及display属性介绍
2013/07/04 Javascript
javascript创建和存储cookie示例
2014/01/07 Javascript
jquery实现页面虚拟键盘特效
2015/08/08 Javascript
基于jquery实现表格无刷新分页
2016/01/07 Javascript
Bootstrap中的Panel和Table全面解析
2016/06/13 Javascript
BootStrap初学者对弹出框和进度条的使用感觉
2016/06/27 Javascript
JS传值出现中文参数乱码的解决方法
2016/06/30 Javascript
使用jquery.qrcode.js生成二维码插件
2016/10/17 Javascript
JS简单实现移动端日历功能示例
2016/12/28 Javascript
浅析JS中什么是自定义react数据验证组件
2018/10/19 Javascript
深入理解JavaScript 中的执行上下文和执行栈
2018/10/23 Javascript
vue项目部署到nginx/tomcat服务器的实现
2019/08/26 Javascript
layui下拉列表select实现可输入查找的方法
2019/09/28 Javascript
Python标准库06之子进程 (subprocess包) 详解
2016/12/07 Python
使用DataFrame删除行和列的实例讲解
2018/04/08 Python
python 3.6.2 安装配置方法图文教程
2018/09/18 Python
对Python subprocess.Popen子进程管道阻塞详解
2018/10/29 Python
Python空间数据处理之GDAL读写遥感图像
2019/08/01 Python
3种方式实现瀑布流布局小结
2019/09/05 HTML / CSS
预订奥兰多和佛罗里达州公园门票:FloridaTix
2018/01/03 全球购物
屈臣氏泰国官网:Watsons TH
2021/02/23 全球购物
Java里面有没有全局变量?为什么?
2015/02/06 面试题
自荐书封面下载
2013/11/29 职场文书
大专生自我评价
2014/01/28 职场文书
献爱心大型公益活动策划方案
2014/09/15 职场文书
2014年商场工作总结
2014/11/22 职场文书
《自然之道》读后感3篇
2019/12/17 职场文书
用javascript制作qq注册动态页面
2021/04/14 Javascript
基于Python和openCV实现图像的全景拼接详细步骤
2021/10/05 Python
Python find()、rfind()方法及作用
2022/12/24 Python