Node.js 天生異步和事件驅(qū)動(dòng),非常適合處理 I/O 相關(guān)的任務(wù)。如果你在處理應(yīng)用中 I/O 相關(guān)的操作,你可以利用 Node.js 中的流(stream)。因此,我們先具體看看流,理解一下它們是怎么簡(jiǎn)化 I/O 操作的吧。
流是什么
流是 unix 管道,讓你可以很容易地從數(shù)據(jù)源讀取數(shù)據(jù),然后流向另一個(gè)目的地。
簡(jiǎn)單來說,流不是什么特別的東西,它只是一個(gè)實(shí)現(xiàn)了一些方法的 EventEmitter 。根據(jù)它實(shí)現(xiàn)的方法,流可以變成可讀流(Readable),可寫流(Writable),或者雙向流(Duplex,同時(shí)可讀可寫)。
可讀流能讓你從一個(gè)數(shù)據(jù)源讀取數(shù)據(jù),而可寫流則可以讓你往目的地寫入數(shù)據(jù)。
如果你已經(jīng)用過 Node.js,你很可能已經(jīng)遇到過流了。
例如,在一個(gè) Node.js 的 HTTP 服務(wù)器里面, request 是一個(gè)可讀流, response 是一個(gè)可寫流。
你也可能用過 fs 模塊,它能幫你處理可讀可寫流。
現(xiàn)在讓你學(xué)一些基礎(chǔ),理解不同類型的流。本文會(huì)討論可讀流和可寫流,雙向流超出了本文的討論范圍,我們不作討論。
可讀流 (Readable Streams)
我們可以用可讀流從一個(gè)數(shù)據(jù)源中讀取數(shù)據(jù),這個(gè)數(shù)據(jù)源可以是任何東西,例如系統(tǒng)中的一個(gè)文件,內(nèi)存中的 buffer,甚至是其他流。因?yàn)榱魇?EventEmitter ,它們會(huì)用各種事件發(fā)送數(shù)據(jù)。我們會(huì)利用這些事件來讓流工作。
從流中讀取數(shù)據(jù)
從流中讀取數(shù)據(jù)最好的方式是監(jiān)聽 data 事件,添加一個(gè)回調(diào)函數(shù)。當(dāng)有數(shù)據(jù)流過來的時(shí)候,可讀流會(huì)發(fā)送 data 事件,回調(diào)函數(shù)就會(huì)觸發(fā)??纯聪旅娴拇a片段:
var fs = require('fs');var readableStream = fs.createReadStream('file.txt');var data = '';var readableStream.on('data', function(chunk){ data += chunk;});readableStream.on('end', function(){ console.log(data);});
fs.createReadStream 會(huì)給你一個(gè)可讀流。
最開始的時(shí)候,這個(gè)流不是流動(dòng)態(tài)的。當(dāng)你添加了 data 的事件監(jiān)聽器,加上一個(gè)回調(diào)函數(shù)時(shí),它才會(huì)變成流動(dòng)態(tài)的。在這之后,它就會(huì)讀取一小塊數(shù)據(jù),然后傳到你的回調(diào)函數(shù)里面。
流的實(shí)現(xiàn)者決定了 data 事件的觸發(fā)頻率,例如 HTTP request 會(huì)在讀取到幾 KB 數(shù)據(jù)的時(shí)候觸發(fā) data 事件。 當(dāng)你從一個(gè)文件中讀取數(shù)據(jù)的時(shí)候,你可能會(huì)決定當(dāng)一行被讀完的時(shí)候就觸發(fā) data 事件。
當(dāng)沒有數(shù)據(jù)可讀的時(shí)候 (讀到文件尾部時(shí)),流就會(huì)發(fā)送 end 事件。在上面的例子中,我們監(jiān)聽了這個(gè)事件,當(dāng)讀完文件的時(shí)候,就把數(shù)據(jù)打印出來。
還有另一種讀取流的方式,你只要在讀到文件尾部前不斷調(diào)用流實(shí)例中的 read() 方法就可以了。
var fs = require('fs');var readableStream = fs.createReadStream('file.txt');var data = '';var chunk;readableStream.on('readable', function(){ while ((chunk = readableStream.read()) != null) { data += chunk; }});readableStream.on('end', function(){ console.log(data);});
read() 方法會(huì)從內(nèi)部 buffer 中讀取數(shù)據(jù),當(dāng)沒有數(shù)據(jù)可讀的時(shí)候,它會(huì)返回 null 。
因此,在 while 循環(huán)中我們檢查 read() 是不是返回 null ,當(dāng)它返回 null 的時(shí)候,就終止循環(huán)。
需要注意的是,當(dāng)我們可以從流中讀取數(shù)據(jù)的時(shí)候, readable 事件就會(huì)觸發(fā)。
設(shè)置編碼
默認(rèn)情況下,你從流中讀取到的是 Buffer 對(duì)象。如果你要讀取的是字符串的話,這并不適合你。因此,你可以像下面的例子那樣通過調(diào)用 Readable.setEncoding() 來設(shè)置流的編碼:
var fs = require('fs');var readableStream = fs.createReadStream('file.txt');var data = '';readableStream.setEncoding('utf8');readableStream.on('data', function(chunk){ data += chunk;});readableStream.on('end', function(){ console.log(data);});
上面的例子中,我們把流的編碼設(shè)置成 utf8 ,數(shù)據(jù)就會(huì)被解析成 utf8 ,回調(diào)函數(shù)中的 chunk 就會(huì)是字符串了。
管道 (Piping)
管道是一個(gè)很棒的機(jī)制,你不需要自己管理流的狀態(tài)就可以從數(shù)據(jù)源中讀取數(shù)據(jù),然后寫入到目的地中。我們先看看下面的例子:
var fs = require('fs');var readableStream = fs.createReadStream('file1.txt');var writableStream = fs.createWriteStream('file2.txt');readableStream.pipe(writableStream);
上面的例子利用 pipe() 方法把 file1 的內(nèi)容寫到 file2 中。因?yàn)?pipe() 會(huì)幫你管理數(shù)據(jù)流,你不需要擔(dān)心數(shù)據(jù)流的速度。這讓 pipe() 變得非常簡(jiǎn)潔易用。
需要注意的是, pipe() 會(huì)返回目的地的流,因此你可以很輕易讓多個(gè)流鏈接起來!
鏈接 (Chaining)
假設(shè)有一個(gè)歸檔文件,你想要解壓它。有很多方式可以完成這個(gè)任務(wù)。但最簡(jiǎn)潔的方式是利用管道和鏈接:
var fs = require('fs');var zlib = require('zlib');fs.createReadStream('input.txt.gz') .pipe(zlib.createGunzip()) .pipe(fs.createWriteStream('output.txt'));
首先,我們通過 input.txt.gz 創(chuàng)建了一個(gè)可讀流,然后讓它流 zlib.createGunzip() 流,它會(huì)解壓內(nèi)容。最后,我們添加一個(gè)可寫流把解壓后的內(nèi)容寫到另一個(gè)文件中。
其他方法
我們已經(jīng)討論了一些可讀流中重要的概念了,這里還有一些你需要知道的方法:
1.Readable.pause()
主站蜘蛛池模板:
莱芜市|
宜兰县|
兴城市|
阳山县|
虎林市|
武陟县|
华亭县|
盐池县|
麦盖提县|
邯郸市|
沁水县|
江油市|
建瓯市|
承德县|
万源市|
磐石市|
花垣县|
渭源县|
广安市|
阳城县|
石林|
孝感市|
永兴县|
海南省|
蒲城县|
通化市|
巴青县|
德保县|
廉江市|
济宁市|
宁乡县|
盱眙县|
加查县|
沙雅县|
乡城县|
博客|
佛教|
阿合奇县|
石泉县|
新蔡县|
闽侯县|