理解nodejs的stream和pipe机制的原理和实现


Posted in NodeJs onAugust 12, 2017

前言

前几天别人请教我关于pipe的问题,我发现我虽然用了nodejs很久,但是由于每次用的不多所以经常回避stream的使用,导致一直不熟,现在重新学习整理一下相关知识。

通过nodeschool学习stream

nodeschool有一个stream-adventure教程教导stream的使用,很简单

简单stream进行pipe

首先,我们可以通过管道将输入定位到输出,输入输出可以是控制台或者文件流或者http请求,比如

process.stdin.pipe(process.stdout)
process.stdin.pipe(fs.createWriteStream(path))
fs.createReadStream(path).pipe(process.stdin)

pipe中间进行处理

如果我们想要在管道中间进行处理,比如想将输入的字符串变成大写写到输出里,我们可以使用一些可以作为中间处理的框架,比如through2就很方便

var through2 = require('through2');
var stream = through2(write,end)
process.stdin
  .pipe(stream)
  .pipe(process.stdout);

function write(line,_,next){
  this.push(line.toString().toUpperCase())
  next();
})
function end(done){
  done();
})

stream转化成普通回调

当我们输入是流,而输出是个普通函数,我们需要把输入流转化为普通的buffer,这时可以试用concat-stream库

var concat = require('concat-stream');

var reverseStream=concat(function(text){
  console.log(text.toString().split("").reverse().join(""));
})

process.stdin.pipe(reverseStream)

http server中的流

类似stdin和fs,http由于其特性也适合使用流,所以其自带类似特性

var http = require('http');
var server = http.createServer(function(req,res){
  req.pipe(res);
})

既作为输入也作为输出的流

request框架实现了如下功能,将一个流pipe到request请求中,然后将流的内容发给服务器,然后返回作为流供其他代码使用,实现如下

var request = require('request');
var r = request.post('http://localhost:8099');
process.stdin.pipe(r).pipe(process.stdout)

分支管道

下边是一个例子,这个例子将输入管道中html包含loud class的元素放入另一个管道进行大写操作,然后最后合并成输出

var trumpet = require('trumpet');
var through2 = require('through2');
var fs = require('fs');
var tr = trumpet();
var stream = tr.select('.loud').createStream();
var upper = through2(function(buf,_,next){
  this.push(buf.toString().toUpperCase());
  next();
})
stream.pipe(upper).pipe(stream);
process.stdin.pipe(tr).pipe(process.stdout);

合并输入输出stream例子

合并后的输入输出可像前文request一样使用,下边这个例子实现了使用流的方式进行子进程调用

var spawn = require('child_process').spawn;
var duplexer2 = require('duplexer2');

module.exports = function(cmd, args){
  var c = spawn(cmd,args)
  return duplexer2(c.stdin,c.stdout)
}

总结

通过上边的例子,可以知道stream应该还有如何合并等更复杂的应用方式。总之整体上符合如下特性:

  • Stream分为readable、writeble
  • Stream通过pipe方法控制流向
  • httpServer和httpClient和file system和process.stdin\out\err通常可以作为stream
  • Stream可以被on(event)转化为普通的变量,普通变量可以被write转换成stream
  • Stream自身可以被拆分、合并、过滤

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持三水点靠木。

NodeJs 相关文章推荐
Nodejs极简入门教程(一):模块机制
Oct 25 NodeJs
nodejs和php实现图片访问实时处理
Jan 05 NodeJs
NodeJS学习笔记之Module的简介
Mar 24 NodeJs
NodeJS处理Express中异步错误
Mar 26 NodeJs
nodejs模块nodemailer基本使用-邮件发送示例(支持附件)
Mar 28 NodeJs
详解nodejs操作mongodb数据库封装DB类
Apr 10 NodeJs
ajax +NodeJS 实现图片上传实例
Jun 06 NodeJs
nodejs mysql 实现分页的方法
Jun 06 NodeJs
nodejs中Express与Koa2对比分析
Feb 06 NodeJs
NodeJS如何实现同步的方法示例
Aug 24 NodeJs
nodejs初始化init的示例代码
Oct 10 NodeJs
nodejs实现范围请求的实现代码
Oct 12 NodeJs
Windows下快速搭建NodeJS本地服务器的步骤
Aug 09 #NodeJs
让nodeJS支持ES6的词法----babel的安装和使用方法
Jul 31 #NodeJs
浅谈nodejs中的类定义和继承的套路
Jul 26 #NodeJs
nodejs之get/post请求的几种方式小结
Jul 26 #NodeJs
nodejs前端自动化构建环境的搭建
Jul 26 #NodeJs
nodejs body-parser 解析post数据实例
Jul 26 #NodeJs
深入解析nodejs HTTP服务
Jul 25 #NodeJs
You might like
xajax写的留言本
2006/11/25 PHP
PHP 日期时间函数的高级应用技巧
2009/10/10 PHP
php遍历目录viewDir函数
2009/12/15 PHP
php购物网站支付paypal使用方法
2010/11/28 PHP
php中随机函数mt_rand()与rand()性能对比分析
2014/12/01 PHP
php微信公众平台开发(一) 配置接口
2016/12/06 PHP
PHP读取并输出XML文件数据的简单实现方法
2017/12/22 PHP
JavaScript二维数组实现的省市联动菜单
2014/05/08 Javascript
js兼容火狐获取图片宽和高的方法
2015/05/21 Javascript
nodejs加密Crypto的实例代码
2016/07/07 NodeJs
Angularjs自定义指令实现三级联动 选择地理位置
2017/02/13 Javascript
js实现自定义进度条效果
2017/03/15 Javascript
NodeJS 将文件夹按照存放路径变成一个对应的JSON的方法
2018/10/17 NodeJs
详解element-ui设置下拉选择切换必填和非必填
2019/06/17 Javascript
layui 表单标签的校验方法
2019/09/04 Javascript
微信小程序实现首页弹出广告
2020/12/03 Javascript
在 Django/Flask 开发服务器上使用 HTTPS
2014/07/03 Python
python追加元素到列表的方法
2015/07/28 Python
Python脚本处理空格的方法
2016/08/08 Python
对Python正则匹配IP、Url、Mail的方法详解
2018/12/25 Python
Python设计模式之状态模式原理与用法详解
2019/01/15 Python
使用Django开发简单接口实现文章增删改查
2019/05/09 Python
python实现关闭第三方窗口的方法
2019/06/28 Python
Python缓存技术实现过程详解
2019/09/25 Python
PyTorch和Keras计算模型参数的例子
2020/01/02 Python
Python 将 QQ 好友头像生成祝福语的实现代码
2020/05/03 Python
Python3+selenium配置常见报错解决方案
2020/08/28 Python
Python机器学习工具scikit-learn的使用笔记
2021/01/28 Python
美国葡萄酒网上商店:Martha Stewart Wine Co.
2019/03/17 全球购物
xml有哪些解析技术?区别是什么
2016/04/26 面试题
2015关于重阳节的演讲稿
2015/03/20 职场文书
漂亮妈妈观后感
2015/06/08 职场文书
大学生实习证明
2015/06/16 职场文书
致运动员的广播稿
2015/08/19 职场文书
高中美术教学反思
2016/02/17 职场文书
win10系统xps文件怎么打开?win10打开xps文件的两种操作方法
2022/07/23 数码科技