Buffer

JavaScript 语言自身只有字符串数据类型,没有二进制数据类型。但在处理像TCP流或文件流时,必须使用到二进制数据。

因此在 Node.js中,定义了一个 Buffer 类,该类用来创建一个专门存放二进制数据的缓存区。

Buffer 库为 Node.js 带来了一种存储原始数据的方法,可以让 Node.js 处理二进制数据,每当需要在 Node.js 中处理I/O操作中移动的数据时,就有可能使用 Buffer 库。

原始数据存储在 Buffer 类的实例中。一个 Buffer 类似于一个整数数组,但它对应于 V8 堆内存之外的一块原始内存。

1.Buffer对象
Buffer对象类似于数组,它的元素为16进制的两位数,即0到255的数值。示例代码如下:

 var str = "深入浅出node.js";
var buf = new Buffer(str, 'utf-8');
console.log(buf);
//<Buffer e6 b7 b1 e5 85 a5 e6 b5 85 e5 87 ba 6e 6f 64 65 2e 6a 73>

上述中的中文字在UTF-8编码下占用3个元素,其余各占用一个元素。

2.字符串转换Buffer
字符串转Buffer对项主要是通过构造函数完成的:new Buffer(str,[encoding]);
通过构造函数转换的Buffer对象,存储的只能是一种编码类型。encoding 参数不传参时,默认按utf-8编码进行转码。

3.Buffer转字符串
Buffer对象的toString()可以将Buffer对象转换为字符串:buf.toString([encoding], [start], [end]);
可以设置encoding(默认为utf-8),start ,end这3个参数实现整体或者局部的转换。

Stream

流是一种抽象的数据结构。想象水流,当在水管中流动时,就可以从某个地方(例如自来水厂)源源不断地到达另一个地方(例如洗手池)。

我们也可以把数据看成是数据流,比如你敲键盘的时候,就可以把每个字符依次连起来,看成字符流。

这个流是从键盘输入到应用程序,实际上它还对应着一个名字:标准输入流(stdin)。

Node.js中Stream 有四种流类型:

  • Readable - 可读操作。

  • Writable - 可写操作。

  • Duplex - 可读可写操作.

  • Transform - 操作被写入数据,然后读出结果。

所有的 Stream 对象都是 EventEmitter 的实例。常用的事件有:

  • data - 当有数据可读时触发。

  • end - 没有更多的数据可读时触发。

  • error - 在接收和写入过程中发生错误时触发。

  • finish - 所有数据已被写入到底层系统时触发。

读取型:

对数据流进行一次性读取:

readstream1.js

 'use strict';
//引入fs模块
var fs = require('fs'); // 创建一个流:
var rs = fs.createReadStream(__dirname+'/sample.txt', 'utf-8');
var data=""; //data事件表示流的数据已经可以读取,此时一次性进行读取。
rs.on('data', function (chunk) {
console.log('Data')
console.log(chunk);
data+=chunk;
}); //end事件表示这个流已经到末尾,无数据可以读取。
rs.on('end', function () {
console.log('End');
console.log(data);
}); //error事件表示出错了。
rs.on('error', function (err) {
console.log('ERROR: ' + err);
});
26 console.log('程序执行完毕');

当对流进行切割,即限制读取长度,此时将会出现乱码,为避免乱码出现,将采用以下方式进行流操作:

readstream2.js

'use strict';
//引入fs模块
var fs = require('fs'); // 创建一个流,将文件可读流的每次读取的Buffer长度限制为11:
var rs = fs.createReadStream(__dirname+'/sample.txt',{highWaterMark: 11});
var chunks = [];
var size = 0;
var i=1; //data事件表示流的数据已经可以读取,此时将进行多次读取,需要将流进行连接。
rs.on('data', function (chunk) {
console.log('Data读取第'+i+++'次');
chunks.push(chunk);
size += chunk.length;
}); //end事件表示这个流已经到末尾,无数据可以读取。
rs.on('end', function () {
console.log('End');
//Buffer.concat()方法封装了从小Buffer对象向大Buffer对象的复制过程。
let buf =Buffer.concat(chunks, size);
console.log(buf);
let str = buf.toString('utf-8');
console.log(str);
}); //error事件表示出错了。
rs.on('error', function (err) {
console.log('ERROR: ' + err);
});
 
 
使用文件描述符来读取文件,以下为异步模式下读取文件的语法格式:
fs.read(fd, buffer, offset, length, position, callback)

参数使用说明如下:

  • fd - 通过 fs.open() 方法返回的文件描述符。

  • buffer - 数据写入的缓冲区。

  • offset - 缓冲区写入的写入偏移量。

  • length - 要从文件中读取的字节数。

  • position - 文件读取的起始位置,如果 position 的值为 null,则会从当前文件指针的位置读取。

  • callback - 回调函数,有三个参数err, bytesRead, buffer,err 为错误信息, bytesRead 表示读取的字节数,buffer 为缓冲区对象。

readfile.js

 var fs = require('fs');
var buffer = new Buffer(1024); //设置缓冲区 console.log('即将打开文件');
fs.open(__dirname+'/sample.txt', 'r+', function (err, fd) {
if (err) {
console.error(err);
}
console.log('文件打开成功'); //fs.read(fd, buffer, offset, length, position, callback)
fs.read(fd, buffer, 0, buffer.length, 0, function (err, bytes) { if (err) {
console.log(err);
}
console.log(bytes + "字节被读取:"); // 仅输出读取的字节
if (bytes > 0) {
console.log(buffer.slice(0, bytes).toString());
} //关闭文件
fs.close(fd, function(err){
if (err){
console.log(err);
}
console.log("文件关闭成功");
});
})
})

写入型:

writestream.js

 'use strict';

 //引入fs模块
var fs = require('fs'); //写入文本数据
var ws1 = fs.createWriteStream(__dirname+'/output1.txt', 'utf-8');
//写数据
ws1.write('使用Stream写入文本数据...\n');
ws1.write('END.');
// 标记文件末尾
ws1.end();
// 处理流事件 --> data, end, and error
ws1.on('finish',function(){
console.log('写入完成');
});
ws1.on('error', function(err){
console.log(err.stack);
}); //写入二进制数据
var ws2 = fs.createWriteStream(__dirname+'/output2.txt');
ws2.write(new Buffer('使用Stream写入二进制数据...\n', 'utf-8'));
ws2.write(new Buffer('END.', 'utf-8'));
ws2.end();
ws2.on('finish',function(){
console.log('写入完成');
});
ws2.on('error', function(err){
console.log(err.stack);
}); console.log("程序执行完毕");

注:该模块的更多使用方法:http://www.runoob.com/nodejs/nodejs-fs.html

Buffer和Stream的更多相关文章

  1. 理解 nodeJS 中的 buffer,stream

    在Node.js开发中,当遇到 buffer,stream,和二进制数据处理时,你是否像我一样,总是感到困惑?这种感觉是否会让你认为不了解它们,以为它们不适合你,认为而这些是Node.js作者们的事情 ...

  2. 11慕课网《进击Node.js基础(一)》Buffer和Stream

    Buffer 用来保存原始数据 (logo.png) 以下代码读取logo.png为buffer类型 然后将buffer转化为string,新建png 可以将字符串配置: data:image/png ...

  3. nodejs stream & buffer 互相转换

    stream 转 buffer function streamToBuffer(stream) { return new Promise((resolve, reject) => { let b ...

  4. Node.js:Buffer浅谈

    Javascript在客户端对于unicode编码的数据操作支持非常友好,但是对二进制数据的处理就不尽人意.Node.js为了能够处理二进制数据或非unicode编码的数据,便设计了Buffer类,该 ...

  5. IRandomAccessStream, IBuffer, Stream, byte[] 之间相互转换

    /* * 用于实现 IRandomAccessStream, IBuffer, Stream, byte[] 之间相互转换的帮助类 */ using System;using System.IO;us ...

  6. Using Live555 to Stream Live Video from an IP camera connected to an H264 encoder

    http://stackoverflow.com/questions/27279161/using-live555-to-stream-live-video-from-an-ip-camera-con ...

  7. 一个轻巧高效的多线程c++stream风格异步日志(一)

    一个轻巧高效的多线程c++stream风格异步日志 一个轻巧高效的多线程c++stream风格异步日志 前言 功能需求 性能需求 Logger实现 LogStream类 Logger类 LogStre ...

  8. c# Stream to File的知识点

    个人倾向使用File.WriteAllByte写入文件: //Stream to File MemoryStream ms=...Stream; ms.Position = ; byte[] buff ...

  9. 流Stream

    System.IO 提供了一个抽象类Stream , Stream类 支持对字节的读写操作.所谓的“流”,指的是Stream,也就是所谓的一个文件区.这个文件区中存储着的信息可以是在内存中,也可以是在 ...

随机推荐

  1. 【译】第20节---数据注解-InverseProperty

    原文:http://www.entityframeworktutorial.net/code-first/inverseproperty-dataannotations-attribute-in-co ...

  2. HDU 4557 Tree(可持久化字典树 + LCA)

    http://acm.hdu.edu.cn/showproblem.php?pid=4757 题意: 给出一棵树,每个结点有一个权值,现在有多个询问,每次询问包含x,y,z三个数,求出在x到y的路径上 ...

  3. android获取屏幕宽度和高度

    1. WindowManager wm = (WindowManager) getContext() .getSystemService(Context.WINDOW_SERVICE); int wi ...

  4. 【Python】【IO】

    # [[IO]] # [文件读写] '''读写文件是最常见的IO操作.Python内置了读写文件的函数,用法和C是兼容的.读写文件前,我们先必须了解一下,在磁盘上读写文件的功能都是由操作系统提供的,现 ...

  5. Centos6.8安装redis(一)

    最近有在学习会话共享的配置,其中一种呢是 nginx+redis+tomcat 的会话共享配置,在记录此会话共享配置之前呢先记录下redis等的安装.这篇先简单记录下redis的安装,是其中一种方式, ...

  6. OpenModelica 在特定目录下生成仿真结果文件

    OMEdit的仿真结果文件存放在:C:\Users\***\AppData\Local\Temp\OpenModelica\OMEdit 可以在仿真时设置仿真结果文件名,可使用用绝对路径或相对路径 使 ...

  7. 百万并发中间件系统的内核设计看Java并发性能优化

    “ 这篇文章,给大家聊聊一个百万级并发的中间件系统的内核代码里的锁性能优化. 很多同学都对Java并发编程很感兴趣,学习了很多相关的技术和知识.比如volatile.Atomic.synchroniz ...

  8. nRF52832的SAADC

    SAADC部分思维导图 1ADC原理 1.1主要特点 1)8/10/12分辨率,使用过采样可达到14位分辨率 2)多达8个通道 单端输入时使用1个通道,2个通道可组成差分输入 单端和差分输入时均可配置 ...

  9. Asp.net core 学习笔记 ( ef core )

    更新 : 2018-11-26 这里记入一下关于 foreignKey cascade action 默认情况下如果我们使用 data annotation required + foreginkey ...

  10. 记录结果再利用的"动态规划"

    2018-09-24 15:01:37 动态规划(DP: Dynamic Programming)是算法设计方法之一,在程序设计竞赛中经常被选作题材.在此,我们考察一些经典的DP问题,来看看DP究竟是 ...