”工欲善其事,必先利其器。“—孔子《论语.录灵公》
首页 > 编程 > 流:Node.js

流:Node.js

发布于2024-11-01
浏览:323

Streams: Node.js

Node.js 中的流完整指南

Node.js 中的流是一种以连续方式处理数据读写的强大方法。它们使您能够有效地处理数据,特别是在处理大量信息或 I/O 操作时。本指南将介绍流的类型、如何使用它们以及实际示例,以帮助您了解流在 Node.js 中的工作原理。

什么是流?

流是允许您以连续方式从源读取数据或将数据写入目标的对象。它们非常适合逐个处理数据,而不是一次读取或写入整个文件或缓冲区。这在处理大型数据集时特别有用,因为它可以显着减少内存使用量。

关键概念

  • 流动模式:数据自动从源流向目的地。
  • 暂停模式:需要手动从源读取数据。
  • 可读流:可以从中读取数据的流。
  • 可写流:可以写入数据的流。
  • 双工流:既可以读取也可以写入数据的流。
  • 转换流:在读取或写入数据时修改或转换数据的流。

流的类型

  1. 可读流:这些流允许您读取数据。示例包括 fs.createReadStream() 和 http.IncomingMessage.

  2. 可写流:这些流允许您写入数据。示例包括 fs.createWriteStream() 和 http.ServerResponse.

  3. 双工流:这些流可以读取和写入数据。示例包括 TCP 套接字和 net.Duplex.

  4. 转换流:这是一种双工流,可以在读取或写入数据时修改数据。示例包括用于压缩的 zlib.createGzip()。

创建可读流

您可以使用内置的 fs 模块创建可读流来读取文件,或者使用stream.Readable创建自定义可读流。

示例:使用可读流读取文件

const fs = require('fs');

// Create a readable stream
const readableStream = fs.createReadStream('example.txt', { encoding: 'utf8' });

// Handling the 'data' event
readableStream.on('data', (chunk) => {
    console.log('New chunk received:', chunk);
});

// Handling the 'end' event
readableStream.on('end', () => {
    console.log('No more data to read.');
});

示例:自定义可读流

const { Readable } = require('stream');

class MyReadableStream extends Readable {
    constructor(options) {
        super(options);
        this.current = 0;
    }

    _read(size) {
        if (this.current  {
    console.log('Received:', chunk.toString());
});

创建可写流

您可以使用 fs 模块或扩展stream.Writable 类来创建可写流。

示例:使用可写流写入文件

const fs = require('fs');

// Create a writable stream
const writableStream = fs.createWriteStream('output.txt');

// Write data to the stream
writableStream.write('Hello, World!\n');
writableStream.write('Writing to a file using streams.\n');

// End the stream
writableStream.end(() => {
    console.log('Finished writing to file.');
});

示例:自定义可写流

const { Writable } = require('stream');

class MyWritableStream extends Writable {
    _write(chunk, encoding, callback) {
        console.log('Writing:', chunk.toString());
        callback(); // Call when done
    }
}

const myWritableStream = new MyWritableStream();
myWritableStream.write('Hello, World!\n');
myWritableStream.write('Writing to custom writable stream.\n');
myWritableStream.end();

使用双工流

双工流可以同时读取和写入数据。一个常见的用例是 TCP 套接字。

示例:创建双工流

const { Duplex } = require('stream');

class MyDuplexStream extends Duplex {
    _read(size) {
        this.push('Data from duplex stream\n');
        this.push(null); // No more data
    }

    _write(chunk, encoding, callback) {
        console.log('Received:', chunk.toString());
        callback();
    }
}

const myDuplexStream = new MyDuplexStream();
myDuplexStream.on('data', (chunk) => {
    console.log('Reading:', chunk.toString());
});

// Write to the duplex stream
myDuplexStream.write('Hello, Duplex!\n');
myDuplexStream.end();

使用转换流

转换流对于修改流经流的数据非常有用。例如,您可以使用转换流来压缩数据。

示例:创建转换流

const { Transform } = require('stream');

class MyTransformStream extends Transform {
    _transform(chunk, encoding, callback) {
        const upperChunk = chunk.toString().toUpperCase();
        this.push(upperChunk);
        callback();
    }
}

const myTransformStream = new MyTransformStream();
myTransformStream.on('data', (chunk) => {
    console.log('Transformed:', chunk.toString());
});

// Pipe data through the transform stream
process.stdin.pipe(myTransformStream).pipe(process.stdout);

管道流

流的强大功能之一是将它们通过管道连接在一起的能力。管道允许您将可读流连接到可写流,这使得传输数据变得容易。

示例:管道流

const fs = require('fs');

// Create a readable stream
const readableStream = fs.createReadStream('input.txt');

// Create a writable stream
const writableStream = fs.createWriteStream('output.txt');

// Pipe the readable stream to the writable stream
readableStream.pipe(writableStream);

writableStream.on('finish', () => {
    console.log('Data has been written to output.txt');
});

Node.js 中的流事件

1. 可读流事件

可读流发出几个重要事件来帮助您管理数据流:

  • data:当一大块数据可供读取时发出。
  • end:当没有更多数据可供读取时发出。
  • error:读取过程中发生错误时发出。
  • close:当流和任何底层资源(如文件描述符)已关闭时发出。

示例:可读流事件

const fs = require('fs');

const readableStream = fs.createReadStream('example.txt');

readableStream.on('data', (chunk) => {
    console.log('Received chunk:', chunk.toString());
});

readableStream.on('end', () => {
    console.log('No more data to read.');
});

readableStream.on('error', (err) => {
    console.error('Error occurred:', err);
});

readableStream.on('close', () => {
    console.log('Stream closed.');
});

2. 可写流事件

可写流也会发出几个事件:

  • drain:当流已满后准备好接受更多数据时发出。
  • finish:当所有数据都写入流并且调用了 end() 方法时发出。
  • error:写入过程中发生错误时发出。
  • close:当流和任何底层资源关闭时发出。

示例:可写流事件

const fs = require('fs');

const writableStream = fs.createWriteStream('output.txt');

writableStream.on('finish', () => {
    console.log('All data has been written to output.txt');
});

writableStream.on('error', (err) => {
    console.error('Error occurred:', err);
});

// Writing data
writableStream.write('Hello, World!\n');
writableStream.write('Writing to a file using streams.\n');
writableStream.end(); // Call end to finish the writing process

3. 转换流事件

转换流从可读流和可写流继承事件,并且它们发出:

  • data:当转换后的块可供读取时发出。
  • end:当没有更多数据可供转换时发出。
  • error:转换过程中发生错误时发出。
  • finish:当所有数据都已处理并写入时发出。

示例:转换流事件

const { Transform } = require('stream');

class MyTransformStream extends Transform {
    _transform(chunk, encoding, callback) {
        const upperChunk = chunk.toString().toUpperCase();
        this.push(upperChunk);
        callback();
    }
}

const myTransformStream = new MyTransformStream();

myTransformStream.on('data', (chunk) => {
    console.log('Transformed chunk:', chunk.toString());
});

myTransformStream.on('end', () => {
    console.log('No more data to transform.');
});

myTransformStream.on('error', (err) => {
    console.error('Error occurred:', err);
});

// Write data to the transform stream
myTransformStream.write('Hello, World!\n');
myTransformStream.write('Transforming this text.\n');
myTransformStream.end(); // End the stream

活动概要

  • 可读流:数据、结束、错误、关闭
  • 可写流:排出、完成、错误、关闭
  • Transform Streams:从可读流和可写流继承数据、结束、错误、完成

结论

Node.js 中的流提供了一种强大且高效的方式来连续处理数据。它们允许您逐段读取和写入数据,这使得它们对于 I/O 操作和处理大型数据集特别有用。了解如何创建和使用不同类型的流,以及如何处理事件,将帮助您在 Node.js 中构建更高效、可扩展的应用程序。

无论您是创建可读、可写、双工还是转换流,流 API 的灵活性都允许您以最适合您的应用程序需求的方式处理数据。

版本声明 本文转载于:https://dev.to/harshm03/streams-nodejs-2j32?1如有侵犯,请联系[email protected]删除
最新教程 更多>
  • 了解异步 JavaScript
    了解异步 JavaScript
    JavaScript 是一种单线程语言,这意味着它一次只能做一件事。然而,Web 应用程序通常需要执行从服务器获取数据等任务,这可能需要一些时间。如果 JavaScript 必须等待每个任务完成才能继续,那么您的 Web 应用程序将会变得缓慢且无响应。这就是异步(async)JavaScript发挥...
    编程 发布于2024-11-07
  • 您应该避免的错误(以及如何修复它们)
    您应该避免的错误(以及如何修复它们)
    作为一名 React 开发人员,很容易陷入某些编码模式,这些模式一开始看起来很方便,但最终可能会导致问题。在这篇博文中,我们将探讨 5 个常见的 React 错误,并讨论如何避免它们,确保您的代码保持高效、可维护和可扩展。 1. 滥用关键道具 错误: {myList.map((ite...
    编程 发布于2024-11-07
  • 如何在 PHP 中访问 JavaScript 变量值?
    如何在 PHP 中访问 JavaScript 变量值?
    在 PHP 中使用 JavaScript 变量值使用同时涉及 JavaScript 和 PHP 的 Web 应用程序时,通常需要在两者之间交换数据两种语言。然而,由于语言的执行环境不同,直接在 PHP 中访问 JavaScript 变量是不可能的。PHP 在服务器端执行,而 JavaScript 在...
    编程 发布于2024-11-07
  • Popver API VS 对话框模态:相同但不同
    Popver API VS 对话框模态:相同但不同
    我在阅读一些科技新闻博客时偶然发现标题 Popover API 登陆 Baseline。我很困惑,在我最近深入前端开发期间,我最近很难习惯在 HTML 中使用 Elements。在浏览博客时,我一直对到目前为止我如何使用该元素感到困惑。 长话短说 选择: 需要用户焦点的模态弹出窗口...
    编程 发布于2024-11-07
  • Go中不嵌入结构体可以实现方法继承吗?
    Go中不嵌入结构体可以实现方法继承吗?
    嵌入式结构:方法继承的探索理解 Go 中的方法继承In在 Go 中,将方法从一种类型继承到另一种类型的能力主要是通过嵌入结构来实现的。此技术涉及将一个结构嵌入另一个结构,允许外部结构访问和利用嵌入结构的方法。嵌入结构的示例考虑以下内容代码片段:type Properties map[string]i...
    编程 发布于2024-11-07
  • 如何在 PHP 中的 Foreach 循环中检索数组键
    如何在 PHP 中的 Foreach 循环中检索数组键
    在 Foreach 循环期间检索数组键:PHP在 PHP 中使用数组时,通常需要检索其中的键和值foreach 循环。 key() 函数提供了一种在迭代期间访问当前键的便捷方法。但是,在某些情况下,它可能不会产生所需的结果。考虑以下代码,其目的是从示例数组生成 HTML 表:foreach($sam...
    编程 发布于2024-11-07
  • 在 JavaScript 中创建对象的方法
    在 JavaScript 中创建对象的方法
    介绍 在 JavaScript 中创建对象的方法有很多种。 对象字面量 Object() 构造函数 Object.create() 构造函数 ES6 类 对象字面量 这可能是在 JavaScript 中创建对象最快、最简单的方法。这也称为对象初始值设定项,是一个由零...
    编程 发布于2024-11-07
  • 如何在 JavaScript 中扩展自定义异常的错误对象?
    如何在 JavaScript 中扩展自定义异常的错误对象?
    扩展 JavaScript 中的错误对象在 JavaScript 中抛出异常时,可能希望扩展内置 Error 对象以创建自定义错误类型。这允许更具体和信息丰富的异常处理。在 JavaScript 中,继承不是通过子类化与 Python 不同,在 Python 中,异常通常是从 Exception 基...
    编程 发布于2024-11-07
  • MySQL如何保证并发操作时数据的完整性?
    MySQL如何保证并发操作时数据的完整性?
    MySQL 并发:确保数据完整性如果您的 MySQL 数据库使用 InnoDB 存储引擎,您可能会担心在执行过程中潜在的并发问题。同时记录更新或插入。本文探讨了 MySQL 如何处理并发以及是否需要在应用程序中加入额外的处理。MySQL 的并发处理MySQL 采用原子性,这意味着单独的 SQL 语句...
    编程 发布于2024-11-07
  • 如何使用 Go 在 SQL 查询中有效连接字符串和值?
    如何使用 Go 在 SQL 查询中有效连接字符串和值?
    在 Go 中有效地制作 SQL 查询在 Go 中将字符串与文本 SQL 查询中的值连接起来可能有点棘手。与 Python 不同,Go 的字符串格式化语法行为不同,导致常见错误,如此处遇到的错误。元组语法错误初始代码片段尝试使用 Python -style 元组,Go 中不支持。这会导致语法错误:qu...
    编程 发布于2024-11-07
  • 为什么 json_encode() 无法使用 Latin1 编码对 MySQL 数据库中的重音字符进行编码?
    为什么 json_encode() 无法使用 Latin1 编码对 MySQL 数据库中的重音字符进行编码?
    MySQL 中 UTF-8 字符的 JSON 编码难题当尝试使用 latin1_swedish_ci 编码从数据库中检索重音字符并使用 json_encode() 将它们编码为 JSON 时,结果可能出乎意料。预期结果(例如“Abord â Plouffe”)会转换为“null”,从而使编码的 JS...
    编程 发布于2024-11-07
  • 如何在 MySQL 中将行转置为列:综合指南
    如何在 MySQL 中将行转置为列:综合指南
    在 MySQL 中将行转换为列在 MySQL 查询中将行转换为列需要在应用程序中执行复杂的查询或手动操作。GROUP_CONCAT 解决方案虽然 GROUP_CONCAT 可以将行转换为单列,但它不提供整个结果集所需的转置。手动查询方法对于更复杂的转置,需要细致的查询,从原始行手动构造每一列。复杂查...
    编程 发布于2024-11-07
  • 如何解决iOS后台模式下未收到GCM通知的问题
    如何解决iOS后台模式下未收到GCM通知的问题
    当应用程序在 iOS 上处于后台模式时未收到 GCM 通知当 iOS 在后台收到通知但不处理时,会出现此问题它们在用户界面中。要解决此问题,请确保您的应用:启用后台推送通知:检查您的应用是否已请求并获得在后台接收推送通知的权限。设置徽章应用程序图标:验证是否在应用程序的“设置”>“通知”部分中选择了...
    编程 发布于2024-11-07
  • 为什么在 Windows 7 中使用 CLASSPATH 时出现 ClassNotFoundException?
    为什么在 Windows 7 中使用 CLASSPATH 时出现 ClassNotFoundException?
    尽管使用 CLASSPATH 环境变量仍解决 java.lang.ClassNotFoundException在 Windows 7 中尝试使用 Java 连接到 MySQL 数据库时,设置 CLASSPATH 环境变量以包含 JDBC 驱动程序 jar 文件的路径似乎无法解决 java.lang....
    编程 发布于2024-11-07
  • 开发人员需要了解免费外汇 API
    开发人员需要了解免费外汇 API
    如果您是一名开发人员,您一定正在寻找可以帮助您更轻松地工作的工具,对吗?免费的外汇 API 就是其中之一!它使您无需支付任何费用即可获取外汇汇率。但是,许多开发人员对这些 API 不太了解。因此,本文旨在解释什么是免费外汇 API、它为何有用以及如何为您的项目选择一个 API。 什么是免费外汇 A...
    编程 发布于2024-11-07

免责声明: 提供的所有资源部分来自互联网,如果有侵犯您的版权或其他权益,请说明详细缘由并提供版权或权益证明然后发到邮箱:[email protected] 我们会第一时间内为您处理。

Copyright© 2022 湘ICP备2022001581号-3