Stream 是一个抽象接口,Node 中有很多对象实现了这个接口。例如,对 http 服务器发起请求的 request 对象就是一个 Stream,还有 stdout(标准输出)。
Node.js,Stream 有四种流类型:
- Readable - 可读操作。
- Writable - 可写操作。
- Duplex - 可读可写操作.
- Transform - 操作被写入数据,然后读出结果。
所有的 Stream 对象都是 EventEmitter 的实例。常用的事件有:
- data - 当有数据可读时触发。
- end - 没有更多的数据可读时触发。
- error - 在接收和写入过程中发生错误时触发。
- finish - 所有数据已被写入到底层系统时触发。
从流中读取数据
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19
| var fs = require("fs") var data = ""
var readerStream = fs.createReadStream("a.txt")
readerStream.setEncoding("UTF8")
readerStream.on("data", function (chunk) { data += chunk })
readerStream.on("end", function () { console.log(data) })
console.log("程序执行完毕")
|
写入流
创建 main.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 = "菜鸟教程官网地址:www.runoob.com"
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("程序执行完毕")
|
管道流
管道提供了一个输出流到输入流的机制。通常我们用于从一个流中获取数据并将数据传递到另外一个流中。
![]()
如上面的图片所示,我们把文件比作装水的桶,而水就是文件里的内容,我们用一根管子(pipe)连接两个桶使得水从一个桶流入另一个桶,这样就慢慢的实现了大文件的复制过程。
以下实例我们通过读取一个文件内容并将内容写入到另外一个文件中。
1 2 3 4 5 6 7 8 9
| var fs = require("fs")
var readerStream = fs.createReadStream("input.txt")
var writerStream = fs.createWriteStream("output.txt")
readerStream.pipe(writerStream)
|
这里要注意 .pipe() 是实例方法(可读流调用),只能链式拼接两个流,只自动处理背压,错误、资源清理全都要自己手动写;
stream.pipeline() 是模块顶层函数,官方推荐的安全升级版,支持多流串联、统一错误捕获、自动销毁所有流、杜绝内存泄漏,生产环境优先用 pipeline。
pipeline 的现代 async/await 版本(项目首选)
1 2 3 4 5 6 7 8 9 10 11 12 13 14
| const fs = require('node:fs'); const { pipeline } = require('node:stream/promises');
(async () => { try { await pipeline( fs.createReadStream('input.txt'), fs.createWriteStream('output.txt') ); console.log('复制成功'); } catch (err) { console.error('失败', err); } })();
|
链式流
链式是通过连接输出流到另外一个流并创建多个流操作链的机制。链式流一般用于管道操作。
接下来我们就是用管道和链式来压缩和解压文件。
创建 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("文件压缩完成。")
|
使用 pipeline 的好处是,它可以添加若干个转换器,即输入流经过若干转换后,再进入输出流。如果我们添加的转换器实现了 gzip 功能,那么实际上就可以把输入流自动压缩后进入输出流。
接下来,让我们来解压该文件,创建 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("文件解压完成。")
|
参考
https://www.runoob.com/nodejs/nodejs-stream.html