1. 什么是Buffer
- 缓冲区Buffer是暂时存放输入输出数据的一段内存。
- JS语言没有二进制数据类型,而在处理TCP和文件流的时候,必须要处理二进制数据。
- NodeJS提供了一个Buffer对象来提供对二进制数据的操作
- 是一个表示固定内存分配的全局对象,也就是说要放到缓存区中的字节数需要提前确定
- Buffer好比由一个8位字节元素组成的数组,可以有效的在JavasScript中表示二进制数据
2. 什么是字节
- 字节(Byte)是计算机存储时的一种计量单位,一个字节等于8位二进制数
- 一个位就代表一个0或1,每8个位(bit)组成一个字节(Byte)
- 字节是通过网络传输信息的单位
- 一个字节最大值十进制数是255
2**8-1;
进制
0b 2进制
0x 16进制
0o 8进制
-
将任意进制字符串转换为十进制
- parseInt("11", 2); // 3 2进制转10进制
- parseInt("77", 8); // 63 8进制转10进制
- parseInt("e7", 16); //175 16进制转10进制
-
将10进制转换为其它进制字符串
- (3).toString(2)) // "11" 十进制转2进制
- (17).toString(16) // "11" 十进制转16进制
- (33).toString(32) // "11" 十提制转32进制
3. 定义buffer的三种方式
3.1 通过长度定义buffer
// 创建一个长度为 10、且用 0 填充的 Buffer。
const buf1 = Buffer.alloc(10);
// 创建一个长度为 10、且用 0x1 填充的 Buffer。
const buf2 = Buffer.alloc(10, 1);
// 创建一个长度为 10、且未初始化的 Buffer。
const buf3 = Buffer.allocUnsafe(10);
3.2 通过数组定义buffer
// 创建一个包含 [0x1, 0x2, 0x3] 的 Buffer。
const buf4 = Buffer.from([1, 2, 3]);
正常情况下为0-255之间;
3.3 字符串创建
const buf5 = Buffer.from('张熙沐枫');
4.buffer常用方法
4.1 buf.fill(value[, offset[, end]][, encoding])
手动初始化,擦干净桌子,将buffer内容清0
buffer.fill(0);
4.2 write方法
buf.write(string[, offset[, length]][, encoding])
buffer.write('沐',0,3,'utf8');
buffer.write('枫',3,3,'utf8'); //沐枫
4.3 writeInt8
var buf = new Buffer(4);
buf.writeInt8(0,0);
buf.writeInt8(16,1);
buf.writeInt8(32,2);
buf.writeInt8(48,3);//16*3*/
console.log(buf);
console.log(buf.readInt8(0));
console.log(buf.readInt8(1));
console.log(buf.readInt8(2));
console.log(buf.readInt8(3));
4.3.1 Little-Endian&Big-Endian
不同的CPU有不同的字节序类型,这些字节序是指整数在内存中保存的顺序。
Big-endian:将高序字节存储在起始地址(高位编址)
-
Little-endian:将低序字节存储在起始地址(低位编址)
let buf3 = new Buffer(4); buf3.writeInt16BE(2**8,0); console.log(buf3);//
console.log(buf3.readInt16BE(0)); buf3.writeInt16LE(2**8,2); console.log(buf3);// console.log(buf3.readInt16LE(2));
4.3 toString方法
buf.toString([encoding[, start[, end]]])
buffer.toString('utf8',3,6)
4.4 slice方法 [#](#t134.4 slice方法)
buf.slice([start[, end]])
let newBuf = buffer.slice(0,4);
4.4.1 截取乱码问题
let {StringDecoder} = require('string_decoder');
let sd = new StringDecoder();
let buffer = new Buffer('沐枫');
console.log(sd.write(buffer.slice(0,4)));
console.log(sd.write(buffer.slice(4)));
4.5 copy方法
复制Buffer 把多个buffer拷贝到一个大buffer上
buf.copy(target[, targetStart[, sourceStart[, sourceEnd]]])
let buf5 = Buffer.from('张熙沐枫');
let buf6 = Buffer.alloc(6);
buf5.copy(buf6,0,0,4);
buf5.copy(buf6,3,3,6);
//buf6=沐枫
Buffer.prototype.copy2 = function(targetBuffer,targetStart,sourceStart,sourceEnd){
for(let i=sourceStart;i
4.6 concat方法
Buffer.concat(list[, totalLength])
let buf1 = Buffer.from('沐');
let buf2 = Buffer.from('枫');
let buf3 = Buffer.concat([buf1,buf2],3);
console.log(buf3.toString());
Buffer.concat2 = function (list = [], total = list.reduce((len, item) => len + item.length, 0)) {
if (list.length == 1)
return list[0];
let result = Buffer.alloc(total);
let pos = 0;
for (let bf of list) {
for (let b of bf) {
if (pos < total)
result[pos++] = b;
else
return result;
}
}
return result;
}
Buffer.myConcat = function(list,totalLength=list.reduce((len,item)=>len+item.length,0)){
if(list.length==0)
return list[0];
let newBuffer = Buffer.alloc(totalLength);
let offset = 0;
for(let i=0;itotalLength){
list[i].copy(newBuffer,offset,0,totalLength-offset);
break;
}else{
list[i].copy(newBuffer,offset,0,list[i].length);
offset+=list[i].length;
}
}
return newBuffer;
}
4.7 isBuffer
判断是否是buffer
Buffer.isBuffer
4.8 length
获取字节长度(显示是字符串所代表buffer的长度)
Buffer.byteLength("沐枫");
buffer.length;
6. base64
Base64是网络上最常见的用于传输8Bit字节码的编码方式之一,Base64就是一种基于64个可打印字符来表示二进制数据的方法。
-
Base64要求把每三个8Bit的字节转换为四个6Bit的字节(3_8 = 4_6 = 24),然后把6Bit再添两位高位0,组成四个8Bit的字节,也就是说,转换后的字符串理论上将要比原来的长1/3
const CHARTS = 'ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/'; function transfer(str){ let buf = Buffer.from(str); let result = ''; for(let b of buf){ result += b.toString(2); } return result.match(/(\d{6})/g).map(val=>parseInt(val,2)).map(val=>CHARTS[val]).join(''); } let r = transfer('珠');//54+g