面试官:说说对 Node 中的 Stream 的理解?应用场景?

大数据
流(Stream),是一种数据传输手段,是端到端信息交换的一种方式,是有顺序的,是逐块读取数据、处理内容,用于顺序读取输入或写入输出.

[[404387]]

 本文转载自微信公众号「JS每日一题」,作者灰灰 。转载本文请联系JS每日一题公众号。

一、是什么

流(Stream),是一种数据传输手段,是端到端信息交换的一种方式,是有顺序的,是逐块读取数据、处理内容,用于顺序读取输入或写入输出

在很多时候,流(Stream)是字节流(Byte Steram)的简称,也就是长长的一串字节

除了字节流,还可以有视频流、音频流、数据流

流的独特之处在于,它不像传统的程序那样一次将一个文件读入内存,而是逐块读取数据、处理其内容,而不是将其全部保存在内存中

流可以分成三部分:source、dest、pipe

在source和dest之间有一个连接的管道pipe,它的基本语法是source.pipe(dest),source和dest就是通过pipe连接,让数据从source流向了dest,如下图所示:

二、种类

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

可写流:可写入数据的流。例如 fs.createWriteStream() 可以使用流将数据写入文件

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

在NodeJS中HTTP服务器模块中,request 是可读流,response 是可写流。还有fs 模块,能同时处理可读和可写文件流

可读流和可写流都是单向的,比较容易理解,而另外两个是双向的

双工流

之前了解过websocket通信,是一个全双工通信,发送方和接受方都是各自独立的方法,发送和接收都没有任何关系

如下图所示:

基本代码如下:

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

转换流

转换流的演示图如下所示:

比如一个 babel,把es6转换为es5,我们在左边写入 es6,从右边读取 es5

基本代码如下所示:

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

三、应用场景

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管道把数据流转过去

  1. const fs = require('fs'
  2. const path = require('path'
  3.  
  4. // 两个文件名 
  5. const fileName1 = path.resolve(__dirname, 'data.txt'
  6. const fileName2 = path.resolve(__dirname, 'data-bak.txt'
  7. // 读取文件的 stream 对象 
  8. const readStream = fs.createReadStream(fileName1) 
  9. // 写入文件的 stream 对象 
  10. const writeStream = fs.createWriteStream(fileName2) 
  11. // 通过 pipe执行拷贝,数据流转 
  12. readStream.pipe(writeStream) 
  13. // 数据读取完成监听,即拷贝完成 
  14. readStream.on('end'function () { 
  15.     console.log('拷贝完成'
  16. }) 

一些打包工具的底层操作

目前一些比较火的前端打包构建工具,都是通过node.js编写的,打包和构建的过程肯定是文件频繁操作的过程,离不开stream,如gulp

参考文献

https://xie.infoq.cn/article/1a9695020828460eb3c4ff1fa

 

https://juejin.cn/post/6844903891083984910

 

责任编辑:武晓燕 来源: JS每日一题
相关推荐

2021-06-07 09:41:48

NodeBuffer 网络协议

2021-05-31 10:35:34

TCPWebSocket协议

2021-07-07 08:36:45

React应用场景

2021-07-12 08:35:24

组件应用场景

2021-09-16 07:52:18

算法应用场景

2021-06-10 07:51:07

Node.js循环机制

2021-09-06 10:51:27

TypeScriptJavaScript

2021-06-03 08:14:01

NodeProcessJavaScript

2021-06-01 08:25:06

Node.jsJavaScript运行

2021-06-30 07:19:36

React事件机制

2021-11-04 06:58:32

策略模式面试

2021-08-16 08:33:26

git

2021-11-10 07:47:49

组合模式场景

2021-11-03 14:10:28

工厂模式场景

2021-11-05 07:47:56

代理模式对象

2021-11-09 08:51:13

模式命令面试

2021-06-04 07:55:30

Node Fs 操作

2021-09-08 07:49:34

TypeScript 泛型场景

2021-09-29 07:24:20

场景数据

2021-11-11 16:37:05

模板模式方法
点赞
收藏

51CTO技术栈公众号