理解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 相关文章推荐
基于promise.js实现nodejs的promises库
Jul 06 NodeJs
Nodejs学习笔记之Stream模块
Jan 13 NodeJs
NodeJS学习笔记之MongoDB模块
Jan 13 NodeJs
使用Angular和Nodejs、socket.io搭建聊天室及多人聊天室
Aug 21 NodeJs
NodeJs的优势和适合开发的程序
Aug 14 NodeJs
NodeJS仿WebApi路由示例
Feb 28 NodeJs
nodejs入门教程五:连接数据库的方法分析
Apr 24 NodeJs
nodejs发送http请求时遇到404长时间未响应的解决方法
Dec 10 NodeJs
nodejs简单抓包工具使用详解
Aug 23 NodeJs
nodejs实现聊天机器人功能
Sep 19 NodeJs
Nodejs实现图片上传、压缩预览、定时删除功能
Oct 25 NodeJs
nodejs中内置模块fs,path常见的用法说明
Nov 07 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
用PHP制作静态网站的模板框架
2006/10/09 PHP
php获取字符串前几位的实例(substr返回字符串的子串用法)
2017/03/08 PHP
javascript setTimeout和setInterval 的区别
2009/12/08 Javascript
使用jQuery模板来展现json数据的代码
2010/10/22 Javascript
JS保留两位小数 四舍五入函数的小例子
2013/11/20 Javascript
javascript/jquery获取地址栏url参数的方法
2014/03/05 Javascript
在JavaScript中正确引用bind方法的应用
2015/05/11 Javascript
详细分析JavaScript函数定义
2015/07/16 Javascript
javascript使用 concat 方法对数组进行合并的方法
2016/09/08 Javascript
无阻塞加载js,防止因js加载不了影响页面显示的问题
2016/12/18 Javascript
javascript中递归的两种写法
2017/01/17 Javascript
常用的js方法合集
2017/03/10 Javascript
mpvue写一个CPASS小程序的示例
2018/09/04 Javascript
jquery判断滚动条距离顶部的距离方法
2018/09/05 jQuery
Bootstrap的aria-label和aria-labelledby属性实例详解
2018/11/02 Javascript
如何为vue的项目添加单元测试
2018/12/19 Javascript
javascript定时器的简单应用示例【控制方块移动】
2019/06/17 Javascript
jQuery表单校验插件validator使用方法详解
2020/02/18 jQuery
JS删除对象中某一属性案例详解
2020/09/08 Javascript
vue路由的配置和页面切换详解
2020/09/09 Javascript
使用PYTHON创建XML文档
2012/03/01 Python
Python实现简单登录验证
2016/04/13 Python
PyQt4 treewidget 选择改变颜色,并设置可编辑的方法
2019/06/17 Python
python实现两个经纬度点之间的距离和方位角的方法
2019/07/05 Python
Python实现二叉树的最小深度的两种方法
2019/09/30 Python
pytorch中nn.Conv1d的用法详解
2019/12/31 Python
python读取mysql数据绘制条形图
2020/03/25 Python
杰夫·班克斯男士服装网上商店:Jeff Banks
2019/10/24 全球购物
大学本科毕业生的自我鉴定
2013/11/26 职场文书
安全生产汇报材料
2014/02/17 职场文书
计算机毕业生自荐信范文
2014/03/23 职场文书
机动车交通事故协议书
2015/01/29 职场文书
写给媳妇的检讨书
2015/05/06 职场文书
感谢信
2019/04/11 职场文书
2019西餐厅创业计划书范文!
2019/07/12 职场文书
创业计划书详解
2019/07/19 职场文书