zoukankan      html  css  js  c++  java
  • NodeJS学习笔记 (28)流操作-stream(ok)

    模块概览

    nodejs的核心模块,基本上都是stream的的实例,比如process.stdout、http.clientRequest。

    对于大部分的nodejs开发者来说,平常并不会直接用到stream模块,只需要了解stream的运行机制即可(非常重要)。

    而对于想要实现自定义stream实例的开发者来说,就得好好研究stream的扩展API了,比如gulp的内部实现就大量用到了自定义的stream类型。

    来个简单的例子镇楼,几行代码就实现了读取文件内容,并打印到控制台:

    const fs = require('fs');
    
    fs.createReadStream('./sample.txt').pipe(process.stdout);

    Stream分类

    在nodejs中,有四种stream类型:

    • Readable:用来读取数据,比如 fs.createReadStream()
    • Writable:用来写数据,比如 fs.createWriteStream()
    • Duplex:可读+可写,比如 net.Socket()
    • Transform:在读写的过程中,可以对数据进行修改,比如 zlib.createDeflate()(数据压缩/解压)。

    Readable Stream

    以下都是nodejs中常见的Readable Stream,当然还有其他的,可自行查看文档。

    • http.IncomingRequest
    • fs.createReadStream()
    • process.stdin
    • 其他

    例子一:

    var fs = require('fs');
    
    fs.readFile('./sample.txt', 'utf8', function(err, content){
    	// 文件读取完成,文件内容是 [你好,我是程序猿小卡]
    	console.log('文件读取完成,文件内容是 [%s]', content);
    });

    例子二:

    var fs = require('fs');
    
    var readStream = fs.createReadStream('./sample.txt');
    var content = '';
    
    readStream.setEncoding('utf8');
    
    readStream.on('data', function(chunk){
    	content += chunk;
    });
    
    readStream.on('end', function(chunk){
    	// 文件读取完成,文件内容是 [你好,我是程序猿小卡]
    	console.log('文件读取完成,文件内容是 [%s]', content);
    });

    例子三:

    这里使用了.pipe(dest),好处在于,如果源文件较大,对于降低内存占用有好处。

    var fs = require('fs');
    
    fs.createReadStream('./sample.txt').pipe(process.stdout);

    注意:这里只是原封不动的将内容输出到控制台,所以实际上跟前两个例子有细微差异。可以稍做修改,达到上面同样的效果

    var fs = require('fs');
    
    var onEnd = function(){
    	process.stdout.write(']');	
    };
    
    var fileStream = fs.createReadStream('./sample.txt');
    fileStream.on('end', onEnd)
    
    fileStream.pipe(process.stdout);
    
    process.stdout.write('文件读取完成,文件内容是[');
    
    // 文件读取完成,文件内容是[你好,我是程序猿小卡]


    Writable Stream

    同样以写文件为例子,比如想将hello world写到sample.txt里。

    例子一:

    var fs = require('fs');
    var content = 'hello world';
    var filepath = './sample.txt';
    
    fs.writeFile(filepath, content);

    例子二:

    var fs = require('fs');
    var content = 'hello world';
    var filepath = './sample.txt';
    
    var writeStram = fs.createWriteStream(filepath);
    writeStram.write(content);
    writeStram.end();


    Duplex Stream

    最常见的Duplex stream应该就是net.Socket实例了,在前面的文章里有接触过,这里就直接上代码了,这里包含服务端代码、客户端代码。

    服务端代码:

    var net = require('net');
    var opt = {
    	host: '127.0.0.1',
    	port: '3000'
    };
    
    var client = net.connect(opt, function(){
    	client.write('msg from client');  // 可写
    });
    
    // 可读
    client.on('data', function(data){
        // server: msg from client [msg from client]
    	console.log('client: got reply from server [%s]', data);
    	client.end();
    });

    客户端代码:

    var net = require('net');
    var opt = {
    	host: '127.0.0.1',
    	port: '3000'
    };
    
    var client = net.connect(opt, function(){
    	client.write('msg from client');  // 可写
    });
    
    // 可读
    client.on('data', function(data){
        // lient: got reply from server [reply from server]
    	console.log('client: got reply from server [%s]', data);
    	client.end();
    });


    Transform Stream

    Transform stream是Duplex stream的特例,也就是说,Transform stream也同时可读可写。跟Duplex stream的区别点在于,Transform stream的输出与输入是存在相关性的。

    常见的Transform stream包括zlibcrypto,这里举个简单例子:文件的gzip压缩。

    var fs = require('fs');
    var zlib = require('zlib');
    
    var gzip = zlib.createGzip();
    
    var inFile = fs.createReadStream('./extra/fileForCompress.txt');
    var out = fs.createWriteStream('./extra/fileForCompress.txt.gz');
    
    inFile.pipe(gzip).pipe(out);

    相关链接

    https://nodejs.org/api/stream.html

  • 相关阅读:
    没提供编码格式,读文件时要怎么推测文件具体的编码
    Spring系列.@EnableRedisHttpSession原理简析
    程序员必备画图技能之——流程图
    痞子衡嵌入式:第一本Git命令教程(7.1)- 清理之缓存(stash)
    痞子衡嵌入式:第一本Git命令教程(6)- 日志(log/reflog/gitk)
    痞子衡嵌入式:第一本Git命令教程(5)- 提交(commit/format-patch/am)
    痞子衡嵌入式:第一本Git命令教程(4)- 转移(add/rm/mv)
    痞子衡嵌入式:第一本Git命令教程(3)- 变动(status/diff)
    痞子衡嵌入式:第一本Git命令教程(2)- 连接(remote/clone)
    痞子衡嵌入式:第一本Git命令教程(1)- 准备(init/config/.gitignore)
  • 原文地址:https://www.cnblogs.com/eret9616/p/9197191.html
Copyright © 2011-2022 走看看