一、是什么

stream,是一个数据传输手段,是端到端信息交换的一种方式,而且是有顺序的,是逐块读取数据、处理内容,用于顺序输入或写入输出。
Node.js中很多对象都实现了流,总之它是会冒数据(以Buffer为单位)
它的独特之处在于,它不像传统的程序那样一次将一个文件读入内容后全部保存在内容中,而是逐块读取数据、处理其内容。
流可以分成三部分:source/dest/pipe
sourcedest之间有一个连接的管道pipe,它的基本语法是source.pipe(dest)sourcedest就是通过pipe连接,让数据从source流向了dest,如下图所示:
Node Stream模块 - 图1

二、种类

Node.js,几乎所有地方都使用到了流的概念,分成四个种类:

  • 可写流:可写入数据的流。例如 fs.createWriteStream()可以使用流将数据写入文件
  • 可读流:可读取数据的流。例如fs.createReadStream()可以从文件读取内容
  • 双工流:既可读又可写的流。例如net.Socket
  • 转换流:可以在数据写入和读取时修改或转换数据的流。例如,在文件压缩操作中,可以向文件写入压缩数据,并向文件中读取解压数据。

Node.jshttp服务器模块中,request是可读流,response是可写流。还有 fs模块,能同时处理可读和可写文件流。
可读流和可写流都是单向的,比较容易理解,而另外两个是双向的。

双工流

我们都知道websorket通信是一个全双工通信,发送方和接受方都是各自独立的方法,发送和接收都没有任何关系。
如下图所示:
Node Stream模块 - 图2
基本代码如下:

  1. const { Duplex } = require('stream');
  2. const myDuplex = new Duplex({
  3. read(size) {
  4. // ...
  5. },
  6. write(chunk, encoding, callback) {
  7. // ...
  8. }
  9. });

转换流

转换流的演示图如下:
Node Stream模块 - 图3
除了上述压缩包的例子,还比如一个**babel**,把es6转换为:我们在左边写入es6,从右边读取es5
基本代码如下:

  1. const { Transform } = require('stream');
  2. const myTransform = new Transform({
  3. transform(chunk, encoding, callback) {
  4. // ...
  5. }
  6. });

三、应用场景

stream的应用场景主要是处理IO操作,而http请求和文件操作都属于IO操作。
试想一下,如果一次IO操作过大,硬件的开销就大,而将此次大的IO操作进行分段操作,让数据像水管一样流动,直到流动完成。
常见的场景有:

  • get 请求返回文件给客户端
  • 文件操作

    get 请求返回文件给客户端

    使用stream返回文件,res也是一个stream对象,通过pipe管道将文件数据返回
    1. const server = http.createServer(function (req, res) {
    2. const method = req.method; // 获取请求方法
    3. if (method === 'GET') { // get 请求
    4. const fileName = path.resolve(__dirname, 'data.txt');
    5. let stream = fs.createReadStream(fileName);
    6. stream.pipe(res); // 将 res 作为 stream 的 dest
    7. }
    8. });
    9. server.listen(8000);

    文件操作

    创建一个可读数据流readStream,一个可写数据流writeStream,通过pipe管道把数据流转过去 ```javascript const fs = require(‘fs’) const path = require(‘path’)

// 两个文件名 const fileName1 = path.resolve(dirname, ‘data.txt’) const fileName2 = path.resolve(dirname, ‘data-bak.txt’) // 读取文件的 stream 对象 const readStream = fs.createReadStream(fileName1) // 写入文件的 stream 对象 const writeStream = fs.createWriteStream(fileName2) // 通过 pipe执行拷贝,数据流转 readStream.pipe(writeStream) // 数据读取完成监听,即拷贝完成 readStream.on(‘end’, function () { console.log(‘拷贝完成’) })

```