浅析Node.js的Stream模块中的Readable对象


Posted in Javascript onJuly 29, 2015

我一直都很不愿意扯 nodejs 的流,因为从第一次看到它我就觉得它的设计实在是太恶心了。但是没办法,Stream 规范尚未普及,而且确实有很多东西都依赖了 nodejs 的流来实现的,所以我也只能捏着鼻子硬着头皮来扯一扯这又臭又硬的 nodejs 流对象了。

nodejs 自带了一个叫 stream 的模块,引入它便可以得到一组流对象构造器。现在我只说最简单的 stream.Readable。

其实用过 nodejs 的几乎都接触过 Readable 的实例,只是平时没太在意而已。一个非常典型的例子,http 模块中我们处理每个请求时都会有 req 和 res 对象,req 其实就是一个 Readable 对象。我们可以在这个 req 上以流的形式读到 HTTP 请求的实体部分。

那么问题来了,为什么 http 模块要在此处以流的方式设计呢?或者从另一个维度来问这个问题就是「nodejs 如果获取 POST 请求的内容?」。懂得用搜索引擎的同学肯定可以很容易地找到这么一个答案:监听 data 事件收集数据,在 end 事件中把收集到的数据合并起来。是的,这是解决这个问题的方法。但是为什么它如此设计呢?像 PHP 那样直接就可以取到 POST 内容多好?其实这么设计是有好处的,如果我们接收到的数据是非法的,我可以马上察觉,然后响应并断开连接。这样可以避免一些不必要的传输成本。比如上传图片,也许用户错误地选择了一个很大的可执行文件,我们不需要等到这个文件完全上传完毕,只要一个文件头部的若干字节就能判断一个文件是否是图片了。此处使用流的设计就可以先读出前面的几个字节来使用。

上面提到的 data 事件和 end 事件都是 Readable 的事件,这两个事件分别表示收到数据和数据接收完毕。所以其实我们早已知道了 Readable 的用法,只是很多人不知道它是 Readable 对象而已。

但是上面这两个事件仅仅是对 Readable 的消费者而言的事件。内部是如何把一个数据推送到 Readable 对象里面让 Readable 触发出这些事件的呢?那么它就是 push 方法。下面是一个例子,它创建了一个 Readable 对象,这个对象会流出一个递增的数字(这里使用了 babel-node)

import stream from 'stream';

var r = new stream.Readable;

r.on('data', data => {
 console.log(data + '');
});

r.on('end', data => {
 console.log('end');
});

r._read = () => {
 // console.log('before read');
};

void function callee(i) {
 if(i < 10) {
  r.push(i + ''); // 只能传入字符串或 Buffre 对象
 } else {
  r.push(null); // 当输入一个 null 时表示流传输完成,触发 end 事件 
 }
 setTimeout(callee, 500, i + 1);
}(0);

如果仔细看上面代码就会发现一个很神奇的地方,这个代码覆写了 _read 方法,这是什么鬼?其实我也觉得这是个坑,这个私有命名风格就不吐槽了,为何非要覆写这个方法才算实现它?如果没有覆写这个方法,那么在调用 push 时将抛出异常:

Error: not implemented
  at Readable._read (_stream_readable.js:464:22)
  at Readable.read (_stream_readable.js:341:10)

以上这些便是 Readable 对象的基本用法。但是还有更多坑会踩到,这篇文章只是一个最简单的介绍,让大家学会如何造出一个能输出数据的 Readable 对象而已。至于一些 read 之类的基本方法,反正这些也是不科学的设计之一。

Javascript 相关文章推荐
Javascript判断对象是否相等实现代码
Mar 18 Javascript
解析页面加载与js函数的执行 onload or ready
Dec 12 Javascript
纯css+js写的一个简单的tab标签页带样式
Jan 28 Javascript
分享一则JavaScript滚动条插件源码
Mar 03 Javascript
js兼容pc端浏览器并有多种弹出小提示的手机端浮层控件实例
Apr 29 Javascript
js传值后台中文出现乱码的解决方法
Jun 30 Javascript
JavaScript中清空数组的方法总结
Dec 02 Javascript
jQuery简单绑定单个事件的方法示例
Jun 10 jQuery
OkHttp踩坑随笔为何 response.body().string() 只能调用一次
Jan 08 Javascript
vue+iview+less 实现换肤功能
Aug 17 Javascript
vue.js循环radio的实例
Nov 07 Javascript
package.json中homepage属性的作用详解
Mar 11 Javascript
浅谈Javascript数组的使用
Jul 29 #Javascript
举例讲解Node.js中的Writable对象
Jul 29 #Javascript
浅谈Javascript数组索引
Jul 29 #Javascript
JQ实现新浪游戏首页幻灯片
Jul 29 #Javascript
JavaScript中几种排序算法的简单实现
Jul 29 #Javascript
详解JavaScript中数组的相关知识
Jul 29 #Javascript
javascript+canvas实现刮刮卡抽奖效果
Jul 29 #Javascript
You might like
php格式化日期实例分析
2014/11/12 PHP
PHP实现加密的几种方式介绍
2015/02/22 PHP
PHP常用函数总结(180多个)
2016/12/25 PHP
thinkPHP5框架设置404、403等http状态页面的方法
2018/06/05 PHP
一个小型js框架myJSFrame附API使用帮助
2008/06/28 Javascript
利用jQuery插件扩展识别浏览器内核与外壳的类型和版本的实现代码
2011/10/22 Javascript
Javascript中的回调函数和匿名函数的回调示例介绍
2014/05/12 Javascript
js实现异步循环实现代码
2016/02/16 Javascript
js实现导航栏中英文切换效果
2017/01/16 Javascript
vue.js项目打包上线的图文教程
2017/11/16 Javascript
vue-cli webpack2项目打包优化分享
2018/02/07 Javascript
微信小程序实现弹出菜单
2018/07/19 Javascript
vue中beforeRouteLeave实现页面回退不刷新的示例代码
2019/11/01 Javascript
[00:37]DOTA2上海特级锦标赛 Secert 战队宣传片
2016/03/03 DOTA
[01:55]TI9显影之尘系列 - Evil Geniuses
2019/08/22 DOTA
Python 拷贝对象(深拷贝deepcopy与浅拷贝copy)
2008/09/06 Python
Python实现windows下模拟按键和鼠标点击的方法
2015/03/13 Python
Python之日期与时间处理模块(date和datetime)
2017/02/16 Python
Python字典及字典基本操作方法详解
2018/01/30 Python
浅谈python配置与使用OpenCV踩的一些坑
2018/04/02 Python
linux安装Python3.4.2的操作方法
2018/09/28 Python
python线程的几种创建方式详解
2019/08/29 Python
StubHub巴西:购买和出售您的门票
2016/07/22 全球购物
客户代表自我评价范例
2013/09/24 职场文书
单位单身证明范本
2014/01/11 职场文书
《可爱的动物》教学反思
2014/02/22 职场文书
《赶海》教学反思
2014/04/20 职场文书
预备党员期盼十八届四中全会召开思想汇报
2014/10/17 职场文书
2015年超市工作总结
2015/04/09 职场文书
毕业设计工作总结
2015/08/14 职场文书
2019年二手房买卖合同范本
2019/10/14 职场文书
SQLServer 错误: 15404,无法获取有关 Windows NT 组/用户 WIN-8IVSNAQS8T7\Administrator 的信息
2021/06/30 SQL Server
Pandas 稀疏数据结构的实现
2021/07/25 Python
德劲DE1102数字调谐收音机机评
2022/04/07 无线电
《火纹风花雪月无双》预告“神秘雇佣兵” 紫发剑客
2022/04/13 其他游戏
Java+swing实现抖音上的表白程序详解
2022/06/25 Java/Android