位置: 编程技术 - 正文

浅谈Node.js:理解stream(node.js底层原理)

编辑:rootadmin

推荐整理分享浅谈Node.js:理解stream(node.js底层原理),希望有所帮助,仅作参考,欢迎阅读内容。

文章相关热门搜索词:node.js工作原理,node.js gui,node.js理解,node.js工作原理,node js 原理,nodejs原理和特点,node.js理解,node.js理解,内容如对您有帮助,希望把文章链接给更多的朋友!

Stream在node.js中是一个抽象的接口,基于EventEmitter,也是一种Buffer的高级封装,用来处理流数据。流模块便是提供各种API让我们可以很简单的使用Stream。

流分为四种类型,如下所示:

Readable,可读流 Writable,可写流 Duplex,读写流 Transform,扩展的Duplex,可修改写入的数据

1、Readable可读流

通过stream.Readable可创建一个可读流,它有两种模式:暂停和流动。

在流动模式下,将自动从下游系统读取数据并使用data事件输出;暂停模式下,必须显示调用stream.read()方法读取数据,并触发data事件。

所有的可读流最开始都是暂停模式,可以通过以下方法切换到流动模式:

监听'data'事件 调用stream.resume()方法 调用stream.pipe()方法将数据输出到一个可写流Writable

同样地,也可以切换到暂停模式,有两种方法:

如果没有设置pipe目标,调用stream.pause()方法即可。 如果设置了pipe目标,则需要移除所有的data监听和调用stream.unpipe()方法

在Readable对象中有一个_readableSate的对象,通过该对象可以得知流当前处于什么模式,如下所示:

readable._readableState.flowing = null,没有数据消费者,流不产生数据 readable._readableState.flowing = true,处于流动模式 readable._readableState.flowing = false,处于暂停模式

为什么使用流取数据

对于小文件,使用fs.readFile()方法读取数据更方便,但需要读取大文件的时候,比如几G大小的文件,使用该方法将消耗大量的内存,甚至使程序崩溃。这种情况下,使用流来处理是更合适的,采用分段读取,便不会造成内存的'爆仓'问题。

data事件

在stream提供数据块给消费者时触发,有可能是切换到流动模式的时候,也有可能是调用readable.read()方法且有有效数据块的时候,使用如下所示:

readable事件

当流中有可用数据能被读取时触发,分为两种,新的可用的数据和到达流的末尾,前者stream.read()方法返回可用数据,后者返回null,如下所示:

pause和resume方法

stream.pause()方法让流进入暂停模式,并停止'data'事件触发,stream.resume()方法使流进入流动模式,并恢复'data'事件触发,也可以用来消费所有数据,如下所示:

pipe(destination[, options])方法

pipe()方法绑定一个可写流到可读流上,并自动切换到流动模式,将所有数据输出到可写流,以及做好了数据流的管理,不会发生数据丢失的问题,使用如下所示:

以上介绍了多种可读流的数据消费的方法,但对于一个可读流,最好只选择其中的一种,推荐使用pipe()方法。

2、Writable可写流

所有的可写流都是基于stream.Writable类创建的,创建之后便可将数据写入该流中。

write(chunk[, encoding][, callback])方法

write()方法向可写流中写入数据,参数含义:

chunk,字符串或buffer encoding,若chunk为字符串,则是chunk的编码 callback,当前chunk数据写入磁盘时的回调函数

该方法的返回值为布尔值,如果为false,则表示需要写入的数据块被缓存并且此时缓存的大小超出highWaterMark阀值,否则为true。

使用如下所示:

浅谈Node.js:理解stream(node.js底层原理)

背压机制

如果可写流的写入速度跟不上可读流的读取速度,write方法添加的数据将被缓存,逐渐增多,导致占用大量内存。我们希望的是消耗一个数据,再去读取一个数据,这样内存就维持在一个水平上。如何做到这一点?可以利用write方法的返回值来判断可写流的缓存状态和'drain'事件,及时切换可读流的模式,如下所示:

drain事件

如果Writable.write()方法返回false,则drain事件将会被触发,上面的背压机制已经使用了该事件。

finish事件

在调用stream.end()方法之后且所有缓存区的数据都被写入到下游系统,就会触发该事件,如下所示:

end([chunk][, encoding][, callback])方法

end()方法被调用之后,便不能再调用stream.write()方法写入数据,负责将抛出错误。

3、Duplex读写流

Duplex流同时实现了Readable与Writable类的接口,既是可读流,也是可写流。例如'zlib streams'、'crypto streams'、'TCP sockets'等都是Duplex流。

4、Transform流

Duplex流的扩展,区别在于,Transform流自动将写入端的数据变换后添加到可读端。例如:'zlib streams'、'crypto streams'等都是Transform流。

5、四种流的实现

stream模块提供的API可以让我们很简单的实现流,该模块使用require('stream')引用,我们只要继承四种流中的一个基类(stream.Writable, stream.Readable, stream.Duplex, or stream.Transform),然后实现它的接口就可以了,需要实现的接口如下所示:

| Use-case | Class | Method(s) to implement | | ------------- |-------------| -----| | Reading only | Readable | _read | | Writing only | Writable | _write, _writev | | Reading and writing | Duplex | _read, _write, _writev | | Operate on written data, then read the result | Transform | _transform, _flush |

Readable流实现

如上所示,我们只要继承Readable类并实现_read接口即可,,如下所示:

以上代码使用了四种方法创建一个Readable可读流,必须实现_read()方法,以及用到了readable.push()方法,该方法的作用是将指定的数据添加到读取队列。

Writable流实现

我们只要继承Writable类并实现_write或_writev接口,如下所示(只使用两种方法):

Duplex流实现

实现Duplex流,需要继承Duplex类,并实现_read和_write接口,如下所示:

上面的代码实现了_read()方法,可作为可读流来使用,同时实现了_write()方法,又可作为可写流来使用。

Transform流实现

实现Transform流,需要继承Transform类,并实现_transform接口,如下所示:

上面代码中的_transform()方法,其第一个参数,要么为error,要么为null,第二个参数将被自动转发给readable.push()方法,因此该方法也可以使用如下写法:

Object Mode流实现

我们知道流中的数据默认都是Buffer类型,可读流的数据进入流中便被转换成buffer,然后被消耗,可写流写入数据时,底层调用也将其转化为buffer。但将构造函数的objectMode选择设置为true,便可产生原样的数据,如下所示:

下面利用Transform流实现一个简单的CSS压缩工具,如下所示:

标签: node.js底层原理

本文链接地址:https://www.jiuchutong.com/biancheng/374106.html 转载请保留说明!

上一篇:浅谈Node.js:Buffer模块(浅谈特殊儿童的融合教育论文)

下一篇:浅谈Node.js:fs文件系统模块(node.js中fs模块)

  • 小规模开票超过五百万元,会自动升为一般纳税人吗
  • 税收优惠属于政策吗
  • 税务是怎么回事
  • 土地使用税返还是否征税
  • 盈余公积计提比例必须是10%么
  • 小规模免征增值税政策
  • 私人网银可以转到对公账户吗?
  • 企业理财收入如何确定
  • 利润表没有其他业务收入
  • 运费单独开票的账务处理
  • 房地产公司预分红犯法吗
  • 现金支出的一般程序
  • 增值税汇总纳税政策文件
  • 残保金上年在职工资总额
  • 财务费用手续费有哪些
  • 银行退回服务费怎么写分录
  • 收到跨境分红的会计处理怎么做?
  • 分支机构企业所得税计提和缴纳数额不一致怎么办?
  • 小规模纳税人零申报是每季度报吗
  • 关于转让费的问题
  • 营业执照上能看出来是一般纳税人
  • 怎么查询财务章备案信息
  • 企业的对公支出是什么
  • 小规模纳税人是公司还是个人
  • 外国人可以参加中国境内的党派或政治团体
  • 高新技术企业研发费
  • ami bios怎么设置u盘启动
  • 运输公司租赁车辆怎么做账
  • 计提社保公司承担部分的会计分录
  • win11专业版和家庭中文版区别
  • 文件夹不能打开没有反应
  • 使用u盘安装linux
  • 华为鸿蒙系统网速慢怎么解决
  • 余款退回会计分录
  • 管理费用处理方法包括
  • win11怎么压缩图片画质
  • php curl_multi_init
  • 领用配件计入什么科目
  • 商企预提待付费什么意思
  • 最早的笔记本是512m内存
  • 苏格兰高地什么意思
  • 收到无法支付的押金收入
  • 权重函数公式
  • php 遍历树
  • js前端路由
  • php生成guid
  • PHP中使用什么关键字声明变量的作用域为全局
  • linux中login命令
  • 现金流量表季报本月数是三个月的数据吗
  • 个人劳务所得年终汇算清缴
  • 固定资产一次性折旧账务处理
  • 发放职工福利的会计处理
  • 土地使用权转让合法吗
  • 新准则下交易性金融债券
  • 库存现金余额过大的风险
  • 油费预付卡如何办理
  • 库存商品坏账准备怎么算
  • 固定资产清理的税率是多少
  • 对外投资的会计科目
  • 残保金管理使用办法
  • 主营业务收入和库存商品怎么区分
  • 总公司是一般纳税人分公司是什么
  • 供货商做产品配送怎么做
  • 删除数据库重复
  • 计算机二级考试一年考几次
  • vim入门教程
  • mac安装路径选择
  • 老毛桃u盘启动盘制作工具怎么安装win10系统 老毛桃u盘安装win10系统图文教程
  • winxp不显示桌面图标
  • linux设置用户名和密码
  • win7电脑关机出现0xc0000142是什么原因
  • win7 64位纯净版系统c盘空间显示与实际占用空间不对的解决方法图文教程
  • opengl实现光照效果
  • jquery加载函数
  • centos 开机启动程序
  • jquery.css
  • jquery上滑下滑
  • 煤炭资源税税率选煤税率多少黑龙江
  • 北京车位过户手续费多少
  • 一人可以在多家企业上班吗
  • 免责声明:网站部分图片文字素材来源于网络,如有侵权,请及时告知,我们会第一时间删除,谢谢! 邮箱:opceo@qq.com

    鄂ICP备2023003026号

    网站地图: 企业信息 工商信息 财税知识 网络常识 编程技术

    友情链接: 武汉网站建设