Stream 是一个抽象接口,Node 中有很多对象实现了这个接口。例如,对http 服务器发起请求的request 对象就是一个Stream,
还有stdout(标准输出)。
Node.js,Stream 有四种流类型:
- Readable - 可读操作。
- Writable - 可写操作。
- Duplex - 可读可写操作.
- Transform - 操作被写入数据,然后读出结果。
所有的 Stream 对象都是 EventEmitter 的实例 .常用的事件有:
- data - 当有数据可读时触发。
- end - 没有更多的数据可读时触发。
- error - 在接收和写入过程中发生错误时触发。
- finish - 所有数据已被写入到底层系统时触发。
从流中读取数据
创建 input.txt 文件,内容如下:
1 2
| Node.js 是一个基于 Chrome V8 引擎的 JavaScript 运行环境。Node.js 使用了一个事件驱动、非阻塞式 I/O 的模型,使其 轻量又高效。Node.js 的包管理器 npm,是全球最大的开源库生态系统。
|
创建 stream1.js 文件, 代码如下:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23
| var fs = require("fs"); var data = ''; var readerStream = fs.createReadStream('input.txt'); readerStream.setEncoding('UTF8'); readerStream.on('data', function(chunk) { data += chunk; }); readerStream.on('end',function(){ console.log(data); }); readerStream.on('error', function(err){ console.log(err.stack); }); console.log("end");
|
以上代码执行结果如下:
1 2 3
| end Node.js 是一个基于 Chrome V8 引擎的 JavaScript 运行环境。Node.js 使用了一个事件驱动、非阻塞式 I/O 的模型,使其 轻量又高效。Node.js 的包管理器 npm,是全球最大的开源库生态系统。
|
写入流
创建 stream2.js 文件, 代码如下:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22
| var fs = require("fs"); var data = 'hello out'; var writerStream = fs.createWriteStream('output.txt'); writerStream.write(data,'UTF8'); writerStream.end(); writerStream.on('finish', function() { console.log("写入完成。"); }); writerStream.on('error', function(err){ console.log(err.stack); }); console.log("end");
|
以上程序会将 data 变量的数据写入到 output.txt 文件中。代码执行结果如下:
1 2 3
| node stream2.js end 写入完成。
|
查看 output.txt 文件的内容:
1 2
| cat output.txt hello out
|
管道流
管道提供了一个输出流到输入流的机制。通常我们用于从一个流中获取数据并将数据传递到另外一个流中。
我们把文件比作装水的桶,而水就是文件里的内容,我们用一根管子(pipe)连接两个桶使得水从一个桶流入另一个桶,这样就慢慢
的实现了大文件的复制过程。以下实例我们通过读取一个文件内容并将内容写入到另外一个文件中。
设置 input.txt 文件内容如下:
管道流操作实例
创建 stream3.js 文件, 代码如下:
1 2 3 4 5 6 7 8 9 10 11 12 13
| var fs = require("fs"); var readerStream = fs.createReadStream('input.txt'); var writerStream = fs.createWriteStream('output.txt'); readerStream.pipe(writerStream); console.log("程序执行完毕");
|
代码执行结果如下:
查看 output.txt 文件的内容:
管道流操作实例
链式流
链式是通过连接输出流到另外一个流并创建多个对个流操作链的机制。链式流一般用于管道操作。
接下来我们就是用管道和链式来压缩和解压文件。
创建 compress.js 文件, 代码如下:
1 2 3 4 5 6 7 8 9
| var fs = require("fs"); var zlib = require('zlib'); fs.createReadStream('input.txt') .pipe(zlib.createGzip()) .pipe(fs.createWriteStream('input.txt.gz')); console.log("文件压缩完成。");
|
代码执行结果如下:
执行完以上操作后,我们可以看到当前目录下生成了 input.txt 的压缩文件 input.txt.gz。
接下来,让我们来解压该文件,创建 decompress.js 文件,代码如下:
1 2 3 4 5 6 7 8 9
| var fs = require("fs"); var zlib = require('zlib'); fs.createReadStream('input.txt.gz') .pipe(zlib.createGunzip()) .pipe(fs.createWriteStream('input.txt')); console.log("文件解压完成。");
|
代码执行结果如下: