Node.js 的 Stream 是一种处理流式数据的抽象接口,广泛应用于文件操作、网络通信等场景。
流式数据处理的一个主要优点是可以在数据传输过程中就开始处理数据,而不需要等待整个数据加载完毕,这使得 Node.js 能够高效地处理大量数据,而不会占用过多的内存。
Stream 是一个抽象接口,Node 中有很多对象实现了这个接口。例如,对http 服务器发起请求的 request 对象就是一个 Stream,还有stdout(标准输出)。
Node.js,Stream 有四种流类型:
所有的 Stream 对象都是 EventEmitter 的实例。常用的事件有:
var fs = require("fs");
var data = '';
// 创建可读流
var readerStream = fs.createReadStream('input.txt');
// 设置编码为 utf8。
readerStream.setEncoding('UTF8');
// 处理流事件 --> data, end, and error
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("程序执行完毕");
var fs = require("fs");
var data = 'hello world';
// 创建一个可以写入的流,写入到文件 output.txt 中
var writerStream = fs.createWriteStream('output.txt');
// 使用 utf8 编码写入数据
writerStream.write(data,'UTF8');
// 标记文件末尾
writerStream.end();
// 处理流事件 --> finish、error
writerStream.on('finish', function() {
console.log("写入完成。");
});
writerStream.on('error', function(err){
console.log(err.stack);
});
console.log("程序执行完毕");
双工流同时具有可读和可写的能力。
一个典型的双工流是 TCP 套接字。
const net = require('net');
// 创建一个 TCP 服务器
const server = net.createServer((socket) => {
console.log('Client connected.');
// 读取客户端数据
socket.on('data', (data) => {
console.log('Received data:', data.toString());
});
// 向客户端发送数据
socket.write('Hello, Client!\n');
// 监听关闭事件
socket.on('end', () => {
console.log('Client disconnected.');
});
});
server.listen(3000, () => {
console.log('Server listening on port 3000.');
});
转换流是一种特殊的双工流,可以修改或转换数据。常见的转换流包括压缩和解压缩流。
const zlib = require('zlib');
const fs = require('fs');
// 创建一个可读流
const readableStream = fs.createReadStream('example.txt');
// 创建一个转换流(压缩)
const gzip = zlib.createGzip();
// 创建一个可写流
const writableStream = fs.createWriteStream('example.txt.gz');
// 将可读流管道到转换流,再管道到可写流
readableStream.pipe(gzip).pipe(writableStream);
// 监听完成事件
writableStream.on('finish', () => {
console.log('File compressed successfully.');
});
管道提供了一个输出流到输入流的机制。
通常我们用于从一个流中获取数据并将数据传递到另外一个流中。
var fs = require("fs");
// 创建一个可读流
var readerStream = fs.createReadStream('input.txt');
// 创建一个可写流
var writerStream = fs.createWriteStream('output.txt');
// 管道读写操作
// 读取 input.txt 文件内容,并将内容写入到 output.txt 文件中
readerStream.pipe(writerStream);
console.log("程序执行完毕");
链式是通过连接输出流到另外一个流并创建多个流操作链的机制。
链式流一般用于管道操作。
接下来我们就是用管道和链式来压缩和解压文件。
var fs = require("fs");
var zlib = require('zlib');
// 压缩 input.txt 文件为 input.txt.gz
fs.createReadStream('input.txt')
.pipe(zlib.createGzip())
.pipe(fs.createWriteStream('input.txt.gz'));
console.log("文件压缩完成。");
可读流可以暂停和恢复数据的读取。
const fs = require('fs');
const readableStream = fs.createReadStream('example.txt', 'utf8');
readableStream.on('data', (chunk) => {
console.log('Received chunk:', chunk);
readableStream.pause(); // 暂停读取
setTimeout(() => {
readableStream.resume(); // 恢复读取
}, 1000);
});
可以销毁流,释放资源。
const fs = require('fs');
const readableStream = fs.createReadStream('example.txt', 'utf8');
readableStream.on('data', (chunk) => {
console.log('Received chunk:', chunk);
readableStream.destroy(); // 销毁流
});