详解Nodejs Buffer模块的常用API

本篇文章带大家详细介绍一下nodejs buffer模块的常用api。有一定的参考价值,有需要的朋友可以参考一下,希望对大家有所帮助。

详解Nodejs Buffer模块的常用API

模块概览

Buffer是node的核心模块,开发者可以利用它来处理二进制数据,比如文件流的读写、网络请求数据的处理等。

Buffer的API非常多,本文仅挑选 比较常用/容易理解 的API进行讲解,包括Buffer实例的创建、比较、连接、拷贝、查找、遍历、类型转换、截取、编码转换等。【推荐学习:《nodejs》】

创建

new Buffer(array)Buffer.alloc(length)Buffer.allocUnsafe(length)Buffer.from(array)

通过 new Buffer(array)

// Creates a new Buffer containing the ASCII bytes of the string 'buffer'const buf = new Buffer([0x62, 0x75, 0x66, 0x66, 0x65, 0x72]);

登录后复制

验证下:

var array = 'buffer'.split('').map(function(v){    return '0x' + v.charCodeAt(0).toString(16)});console.log( array.join() );// 输出:0x62,0x75,0x66,0x66,0x65,0x72

登录后复制

通过 Buffer.alloc(length)

var buf1 = Buffer.alloc(10);  // 长度为10的buffer,初始值为0x0var buf2 = Buffer.alloc(10, 1);  // 长度为10的buffer,初始值为0x1

登录后复制

var buf3 = Buffer.allocUnsafe(10);  // 长度为10的buffer,初始值不确定

登录后复制

var buf4 = Buffer.from([1, 2, 3])  // 长度为3的buffer,初始值为 0x01, 0x02, 0x03

登录后复制

通过Buffer.from()

例子一:Buffer.from(array)

// [0x62, 0x75, 0x66, 0x66, 0x65, 0x72] 为字符串 "buffer" // 0x62 为16进制,转成十进制就是 98,代表的就是字母 bvar buf = Buffer.from([0x62, 0x75, 0x66, 0x66, 0x65, 0x72]);console.log(buf.toString());

登录后复制

例子二:Buffer.from(string[, encoding])

通过string创建buffer,跟将buffer转成字符串时,记得编码保持一致,不然会出现乱码,如下所示。

var buf = Buffer.from('this is a tést');  // 默认采用utf8// 输出:this is a téstconsole.log(buf.toString());  // 默认编码是utf8,所以正常打印// 输出:this is a tC)stconsole.log(buf.toString('ascii'));  // 转成字符串时,编码不是utf8,所以乱码

登录后复制

对乱码的分析如下:

var letter = 'é';var buff = Buffer.from(letter);  // 默认编码是utf8,这里占据两个字节 var len = buff.length;  // 2var code = buff[0]; // 第一个字节为0xc3,即195:超出ascii的最大支持范围var binary = code.toString(2);  // 195的二进制:10101001var finalBinary = binary.slice(1);  // 将高位的1舍弃,变成:0101001var finalCode = parseInt(finalBinary, 2);  // 0101001 对应的十进制:67var finalLetter = String.fromCharCode(finalCode);  // 67对应的字符:C// 同理 0xa9最终转成的ascii字符为)// 所以,最终输出为 this is a tC)st

登录后复制

例子三:Buffer.from(buffer)

创建新的Buffer实例,并将buffer的数据拷贝到新的实例子中去。

var buff = Buffer.from('buffer');var buff2 = Buffer.from(buff);console.log(buff.toString());  // 输出:bufferconsole.log(buff2.toString());  // 输出:bufferbuff2[0] = 0x61;console.log(buff.toString());  // 输出:bufferconsole.log(buff2.toString());  // 输出:auffer

登录后复制

buffer比较

buf.equals(otherBuffer)

判断两个buffer实例存储的数据是否相同,如果是,返回true,否则返回false。

// 例子一:编码一样,内容相同var buf1 = Buffer.from('A');var buf2 = Buffer.from('A');console.log( buf1.equals(buf2) );  // true// 例子二:编码一样,内容不同var buf3 = Buffer.from('A');var buf4 = Buffer.from('B');console.log( buf3.equals(buf4) );  // false// 例子三:编码不一样,内容相同var buf5 = Buffer.from('ABC');  // var buf6 = Buffer.from('414243', 'hex');console.log(buf5.equals(buf6));    //true//只要比较的两者内容相同,`buf.equals(otherBuffer)` 就返回true

登录后复制

buf.compare(target[, targetStart[, targetEnd[, sourceStart[, sourceEnd]]]])

同样是对两个buffer实例进行比较,不同的是:

可以指定特定比较的范围(通过start、end指定)

返回值为整数,达标buf、target的大小关系

假设返回值为

0:buf、target大小相同。1:buf大于target,也就是说buf应该排在target之后。-1:buf小于target,也就是说buf应该排在target之前。

看例子,官方的例子挺好的,直接贴一下:

const buf1 = Buffer.from('ABC');const buf2 = Buffer.from('BCD');const buf3 = Buffer.from('ABCD');// Prints: 0console.log(buf1.compare(buf1));// Prints: -1console.log(buf1.compare(buf2));// Prints: -1console.log(buf1.compare(buf3));// Prints: 1console.log(buf2.compare(buf1));// Prints: 1console.log(buf2.compare(buf3));// Prints: [ , ,  ]// (This result is equal to: [buf1, buf3, buf2])console.log([buf1, buf2, buf3].sort(Buffer.compare));

登录后复制

Buffer.compare(buf1, buf2)

跟 buf.compare(target) 大同小异,一般用于排序。直接贴官方例子:

const buf1 = Buffer.from('1234');const buf2 = Buffer.from('0123');const arr = [buf1, buf2];// Prints: [ ,  ]// (This result is equal to: [buf2, buf1])console.log(arr.sort(Buffer.compare));

登录后复制

从Buffer.from([62])谈起

这里稍微研究下Buffer.from(array)。下面是官方文档对API的说明,也就是说,每个array的元素对应1个字节(8位),取值从0到255。

Allocates a new Buffer using an array of octets.

数组元素为数字

首先看下,传入的元素为数字的场景。下面分别是10进制、8进制、16进制,跟预期中的结果一致。

var buff = Buffer.from([62])// // buff[0] === parseInt('3e', 16) === 62

登录后复制

var buff = Buffer.from([062])// // buff[0] === parseInt(62, 8) === parseInt(32, 16) === 50

登录后复制

var buff = Buffer.from([0x62])// // buff[0] === parseInt(62, 16) === 98

登录后复制

数组元素为字符串

再看下,传入的元素为字符串的场景。

0开头的字符串,在parseInt(‘062’)时,可以解释为62,也可以解释为50(八进制),这里看到采用了第一种解释。

字符串的场景,跟parseInt()有没有关系,暂未深入探究,只是这样猜想。TODO(找时间研究下)

var buff = Buffer.from(['62'])// // buff[0] === parseInt('3e', 16) === parseInt('62') === 62

登录后复制

var buff = Buffer.from(['062'])// // buff[0] === parseInt('3e', 16) === parseInt('062') === 62

登录后复制

var buff = Buffer.from(['0x62'])// // buff[0] === parseInt('62', 16) === parseInt('0x62') === 98

登录后复制

数组元素大小超出1个字节

感兴趣的同学自行探究。

var buff = Buffer.from([256])// 

登录后复制

Buffer.from(‘1’)

一开始不自觉的会将Buffer.from(‘1’)[0]跟”1″划等号,其实”1″对应的编码是49。

var buff = Buffer.from('1')  // console.log(buff[0] === 1)  // false

登录后复制

这样对比就知道了,编码为1的是个控制字符,表示 Start of Heading。

console.log( String.fromCharCode(49) )  // '1'console.log( String.fromCharCode(1) )  // 'u0001'

登录后复制

buffer连接:Buffer.concat(list[, totalLength])

备注:个人觉得totalLength这个参数挺多余的,从官方文档来看,是处于性能提升的角度考虑。不过内部实现也只是遍历list,将length累加得到totalLength,从这点来看,性能优化是几乎可以忽略不计的。

var buff1 = Buffer.alloc(10);var buff2 = Buffer.alloc(20);var totalLength = buff1.length + buff2.length;console.log(totalLength);  // 30var buff3 = Buffer.concat([buff1, buff2], totalLength);console.log(buff3.length);  // 30

登录后复制

除了上面提到的性能优化,totalLength还有两点需要注意。假设list里面所有buffer的长度累加和为length

totalLength > length:返回长度为totalLength的Buffer实例,超出长度的部分填充0。totalLength

var buff4 = Buffer.from([1, 2]);var buff5 = Buffer.from([3, 4]);var buff6 = Buffer.concat([buff4, buff5], 5);console.log(buff6.length);  // console.log(buff6);  // var buff7 = Buffer.concat([buff4, buff5], 3);console.log(buff7.length);  // 3console.log(buff7);  // 

登录后复制

拷贝:buf.copy(target[, targetStart[, sourceStart[, sourceEnd]]])

使用比较简单,如果忽略后面三个参数,那就是将buf的数据拷贝到target里去,如下所示:

var buff1 = Buffer.from([1, 2]);var buff2 = Buffer.alloc(2);buff1.copy(buff2);console.log(buff2);  // 

登录后复制

另外三个参数比较直观,直接看官方例子

const buf1 = Buffer.allocUnsafe(26);const buf2 = Buffer.allocUnsafe(26).fill('!');for (let i = 0 ; i 

查找:buf.indexOf(value[, byteOffset][, encoding])

跟数组的查找差不多,需要注意的是,value可能是String、Buffer、Integer中的任意类型。

登录后复制String:如果是字符串,那么encoding就是其对应的编码,默认是utf8。Buffer:如果是Buffer实例,那么会将value中的完整数据,跟buf进行对比。Integer:如果是数字,那么value会被当做无符号的8位整数,取值范围是0到255。

另外,可以通过byteOffset来指定起始查找位置。

直接上代码,官方例子妥妥的,耐心看完它基本就理解得差不多了。

const buf = Buffer.from('this is a buffer');// Prints: 0console.log(buf.indexOf('this'));// Prints: 2console.log(buf.indexOf('is'));// Prints: 8console.log(buf.indexOf(Buffer.from('a buffer')));// Prints: 8// (97 is the decimal ASCII value for 'a')console.log(buf.indexOf(97));// Prints: -1console.log(buf.indexOf(Buffer.from('a buffer example')));// Prints: 8console.log(buf.indexOf(Buffer.from('a buffer example').slice(0, 8)));const utf16Buffer = Buffer.from('u039au0391u03a3u03a3u0395', 'ucs2');// Prints: 4console.log(utf16Buffer.indexOf('u03a3', 0, 'ucs2'));// Prints: 6console.log(utf16Buffer.indexOf('u03a3', -4, 'ucs2'));

登录后复制

写:buf.write(string[, offset[, length]][, encoding])

将sring写入buf实例,同时返回写入的字节数。

参数如下:

string:写入的字符串。offset:从buf的第几位开始写入,默认是0。length:写入多少个字节,默认是 buf.length – offset。encoding:字符串的编码,默认是utf8。

看个简单例子

var buff = Buffer.alloc(4);buff.write('a');  // 返回 1console.log(buff);  // 打印 buff.write('ab');  // 返回 2console.log(buff);  // 打印 

登录后复制

填充:buf.fill(value[, offset[, end]][, encoding])

用value填充buf,常用于初始化buf。参数说明如下:

value:用来填充的内容,可以是Buffer、String或Integer。offset:从第几位开始填充,默认是0。end:停止填充的位置,默认是 buf.length。encoding:如果value是String,那么为value的编码,默认是utf8。

例子:

var buff = Buffer.alloc(20).fill('a');console.log(buff.toString());  // aaaaaaaaaaaaaaaaaaaa

登录后复制

转成字符串: buf.toString([encoding[, start[, end]]])

把buf解码成字符串,用法比较直观,看例子

var buff = Buffer.from('hello');console.log( buff.toString() );  // helloconsole.log( buff.toString('utf8', 0, 2) );  // he

登录后复制

转成JSON字符串:buf.toJSON()

var buff = Buffer.from('hello');console.log( buff.toJSON() );  // { type: 'Buffer', data: [ 104, 101, 108, 108, 111 ] }

登录后复制

遍历:buf.values()、buf.keys()、buf.entries()

用于对buf进行for…of遍历,直接看例子。

var buff = Buffer.from('abcde');for(const key of buff.keys()){    console.log('key is %d', key);}// key is 0// key is 1// key is 2// key is 3// key is 4for(const value of buff.values()){    console.log('value is %d', value);}// value is 97// value is 98// value is 99// value is 100// value is 101for(const pair of buff.entries()){    console.log('buff[%d] === %d', pair[0], pair[1]);}// buff[0] === 97// buff[1] === 98// buff[2] === 99// buff[3] === 100// buff[4] === 101

登录后复制

截取:buf.slice([start[, end]])

用于截取buf,并返回一个新的Buffer实例。需要注意的是,这里返回的Buffer实例,指向的仍然是buf的内存地址,所以对新Buffer实例的修改,也会影响到buf。

var buff1 = Buffer.from('abcde');console.log(buff1);  // var buff2 = buff1.slice();console.log(buff2);  // var buff3 = buff1.slice(1, 3);console.log(buff3);  // buff3[0] = 97;  // parseInt(61, 16) ==> 97console.log(buff1);  // 

登录后复制

TODO

创建、拷贝、截取、转换、查找

buffer、arraybuffer、dataview、typedarray

buffer vs 编码

Buffer.from()、Buffer.alloc()、Buffer.alocUnsafe()

Buffer vs TypedArray

文档摘要

关于buffer内存空间的动态分配

Instances of the Buffer class are similar to arrays of integers but correspond to fixed-sized, raw memory allocations outside the V8 heap. The size of the Buffer is established when it is created and cannot be resized.

相关链接

unicode对照表 https://unicode-table.com/cn/#control-character

字符编码笔记:ASCII,Unicode和UTF-8 http://www.ruanyifeng.com/blog/2007/10/ascii_unicode_and_utf-8.html

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

以上就是详解Nodejs Buffer模块的常用API的详细内容,更多请关注【创想鸟】其它相关文章!

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

发布者:PHP中文网,转转请注明出处:https://www.chuangxiangniao.com/p/2713228.html

(0)
上一篇 2025年3月7日 21:21:34
下一篇 2025年3月6日 00:45:52

AD推荐 黄金广告位招租... 更多推荐

相关推荐

  • Nodejs中使用string_decoder模块将buffer转成string

    本篇文章给大家介绍一下nodejs中使用string_decoder模块将buffer转成string的方法。有一定的参考价值,有需要的朋友可以参考一下,希望对大家有所帮助。 模块简介 string_decoder模块用于将Buffer转成…

    2025年3月7日
    200
  • nodejs设置NODE_ENV时发生错误怎么解决?

    本篇文章给大家介绍一下windows环境中nodejs设置node_env错误的解决方案。有一定的参考价值,有需要的朋友可以参考一下,希望对大家有所帮助。 【推荐学习:《nodejs》】 今天在看node文档的时候,看到《Node.js 开…

    2025年3月7日 编程技术
    200
  • 详解Nodejs中的阻塞和非阻塞

    本篇文章带大家理解一下nodejs中的阻塞和非阻塞。有一定的参考价值,有需要的朋友可以参考一下,希望对大家有所帮助。 推荐学习:《nodejs》 对于阻塞和非阻塞这两个概念大家应该都有一些自己的理解: 再简单说一下,阻塞大部分是由于同步模式…

    2025年3月7日
    200
  • 如何使用Nodejs连接Mysql,实现基本的增删改查操作

    本篇文章给大家介绍一下如何使用nodejs连接mysql,实现基本的增删改查(crud)操作。有一定的参考价值,有需要的朋友可以参考一下,希望对大家有所帮助。 【推荐学习:《nodejs》】 接下来示例代码的主要技术点包括 基础框架 Koa…

    2025年3月7日 编程技术
    200
  • 深入了解调试nodejs程序的方法

    对于开发者来说,在开发应用程序的过程中,往往为了开发方便和解决bug需要借助于编程语言的调试功能。一般来说我们需要借助于强大IDE的调试功能来完成这项工作。nodejs也不例外。 今天我们来详细介绍一下如何调试nodejs程序。 开启nod…

    2025年3月7日 编程技术
    200
  • 浅谈浏览器中本地运行Node.js的方法

    本篇文章给大家介绍一下在浏览器中本地运行node.js的方法。有一定的参考价值,有需要的朋友可以参考一下,希望对大家有所帮助。 【推荐学习:《nodejs 教程》】 一切要从收到一封邮件开始 大早上,我收到一封邮件,StackBlitz说正…

    2025年3月7日 编程技术
    200
  • 一起了解Nodejs和命令行程序

    本篇文章给大家介绍一下nodejs和命令行程序。有一定的参考价值,有需要的朋友可以参考一下,希望对大家有所帮助。 造物无言却有情,每于寒尽觉春生。千红万紫安排著,只待新雷第一声。 —— 清.张维屏 《新雷》 源起 植根于Unix系统环境下的…

    2025年3月7日 编程技术
    200
  • Nodejs可以应用于那些场景?

    本篇文章给大家介绍一些nodejs的应用场景。有一定的参考价值,有需要的朋友可以参考一下,希望对大家有所帮助。 nodejsApache 的多线程高并发模式 优点:支持多线程并发 缺点:阻塞 nodejs什么是线程 线程是可以独立运行的最小…

    2025年3月7日
    200
  • 详解使用Docker给nodejs程序打包的方法

    本篇文章带大家了解一下docker,介绍为什么要使用 docker?如何使用 docker?以及使用docker给nodejs程序打包的方法。有一定的参考价值,有需要的朋友可以参考一下,希望对大家有所帮助。 你听到过这样的对话吗? 这种对话…

    2025年3月7日 编程技术
    200
  • 浅谈Nodejs如何进行多线程处理

    本篇文章给大家介绍一下nodejs进行多线程处理的方法。有一定的参考价值,有需要的朋友可以参考一下,希望对大家有所帮助。 Threads à gogo 是nodejs 的原生模块,使用这个模块可以让nodejs 具备多线程处理功能。【推荐学…

    2025年3月7日
    200

发表回复

登录后才能评论