Node.JS教程10:“流”是Node.js最强大的功能之一

流是Nodejs的高级应用,掌握流的使用,才能真正胜任NodeJS开发。

Nodejs中,流是基于事件的API,用于管理和处理数据,而且效率很好!

什么是流?

流是一个抽象接口,Node 中有很多对象实现了这个接口。

例如,对http 服务器发起请求的request 对象就是一个 Stream,还有stdout(标准输出)等。

流的四种类型

dable - 可读操作。

Writable - 可写操作。

Duplex - 可读可写操作.

Transform - 操作被写入数据,然后读出结果。

流的事件

所有的流对象都是 EventEmitter 的实例,流常用的事件有:

data - 当有数据可读时触发。

end - 没有更多的数据可读时触发。

error - 在接收和写入过程中发生错误时触发。

finish - 所有数据已被写入到底层系统时触发。

什么时候使用流?

举例说明:

1、当使用fs.readFileSync同步读取一个文件的时候,所有的数据会被全部读到内存中,这个操作过程中程序会被阻塞。

2、如果使用fs.readFile,由于它是异步方法,那么阻塞不会发生,但数据仍会被全部读到内存中再处理。

当处理大文件压缩、媒体文件等的时候,无疑会很吃力。那么这时,就是使用流的时候了。

3、流会将分批次的读取适量的内容到缓存区进行操作,而不是一次性读取所有目标内容。

这样,程序对内存的使用量会极大减少、执行性能会提升很多:

举例说明,使用流的优势

先使用node内置的核心模块http实现一个简单的静态web服务器:

这段代码使用非阻塞的fs.readfile的方法。

当被访问时,读取文件内容(代码中读取的是本举程代码)并发送给访问者。

测试访问,效果:

可能说,功能并无问题。但如果被读取的文件test10.js文件非常大呢。就会有效率问题。

这时,可以改用流的方式:

注意图中标识出的内容,是修改的内容:使用流的方式读取,通过管道(pipe)传给res。

执行效果一样,但对内存的使用得到优化,性能得到提升。

同时,代码也更简洁。

流不仅高效优雅,扩展性也更强。比如对上面的代码稍做改动,就可以实现gzip压缩传输数据,可以使网页打开更快。

从浏览器信息中可以看到,内容已启用gzip压缩。