說說node中的可讀流和可寫流的區(qū)別

前言

成都創(chuàng)新互聯(lián)于2013年創(chuàng)立,是專業(yè)互聯(lián)網(wǎng)技術(shù)服務(wù)公司,擁有項(xiàng)目網(wǎng)站設(shè)計(jì)、成都網(wǎng)站建設(shè)網(wǎng)站策劃,項(xiàng)目實(shí)施與項(xiàng)目整合能力。我們以讓每一個(gè)夢想脫穎而出為使命,1280元武城做網(wǎng)站,已為上家服務(wù),為武城各地企業(yè)和個(gè)人服務(wù),聯(lián)系電話:18982081108

nodejs中大量的api與流有關(guān),曾經(jīng)看到公司的一些大神的node代碼,實(shí)現(xiàn)一個(gè)接口只需要pipe一下另一個(gè)java接口就可以了。簡單的一行代碼實(shí)在讓人困惑。作為小白的自己一臉懵逼卻又不敢問,因?yàn)楦静恢缽暮螁柶稹,F(xiàn)在終于通過學(xué)習(xí),也能對(duì)流說出個(gè)123,希望和大家共同交流。

流簡介

流分為緩沖模式和對(duì)象模式,緩沖模式只能處理buffer或字符串,對(duì)象模式可以處理js對(duì)象。流又分為四種類型:可讀流、可寫流、雙工流和轉(zhuǎn)換流。后兩種其實(shí)是對(duì)可讀和可寫流的應(yīng)用。所以我想先聊聊可讀流和可寫流。

可讀流

可讀流有兩種模式,并隨時(shí)可以轉(zhuǎn)換,我們可以通過監(jiān)聽可讀流的事件來操作它。

兩種模式(引用自node中文網(wǎng)的描述):

1、流動(dòng)模式:可讀流自動(dòng)讀取數(shù)據(jù),通過EventEmitter接口的事件盡快將數(shù)據(jù)提供給應(yīng)用。

2、暫停模式:必須顯式調(diào)用stream.read()方法來從流中讀取數(shù)據(jù)片段。

暫停模式切換到流動(dòng)模式的api有:

1、監(jiān)聽“data”事件

2、調(diào)用 stream.resume()方法

3、調(diào)用 stream.pipe()方法將數(shù)據(jù)發(fā)送到可寫流

流動(dòng)模式切換到暫停模式的api有:

1、如果不存在管道目標(biāo),調(diào)用stream.pause()方法

2、如果存在管道目標(biāo),調(diào)用 stream.unpipe()并取消'data'事件監(jiān)聽

可讀流事件:'data','readable','error','close','end'

可寫流

可寫流相對(duì)較為簡單,我們也可以通過監(jiān)聽它的事件來操作它。

可寫流事件: 'close','drain','error','finish','pipe','unpipe'

舉個(gè)栗子

我以一個(gè)簡單的例子描述一個(gè)流最常見的場景,談?wù)剬?duì)這個(gè)過程的理解。例子就是:我要讀取一個(gè)文件,然后把它的內(nèi)容寫到另一個(gè)文件(當(dāng)然是用“流”的api,而不是用‘fs'模塊的api)。

說說node中的可讀流和可寫流的區(qū)別

接下來我解釋一下這張圖:

如上圖,當(dāng)我們創(chuàng)建了一個(gè)可讀流的時(shí)候,readable._readableState.flowing屬性默認(rèn)為null,這時(shí)我們有兩種選擇:

1、監(jiān)聽‘readable'事件,這時(shí)可讀流會(huì)讀取64k(可以在創(chuàng)建可讀流時(shí),通過option參數(shù)中的highWaterMark更改)數(shù)據(jù)到流的緩存區(qū)中,等待你用read方法去讀取并消費(fèi)數(shù)據(jù),當(dāng)你用read方法讀了64k數(shù)據(jù)之后,會(huì)再次觸發(fā)readable事件,直到你讀完了源文件的所有數(shù)據(jù)。記住,之所以叫暫停模式是因?yàn)槿绻悴徽{(diào)用read方法,代碼永遠(yuǎn)會(huì)停在這里,什么事情也不會(huì)發(fā)生了。

2、如果你選擇監(jiān)聽‘data'事件,可讀流會(huì)直接讀取64k數(shù)據(jù)并通過‘data'事件的回掉函數(shù)提供給你消費(fèi),并且這個(gè)過程不會(huì)停止,如果源文件中有很多數(shù)據(jù),會(huì)不停的觸發(fā)‘data'事件,直到全部讀取完成。當(dāng)然,在這個(gè)過程中你隨時(shí)可以通過stream.pause()方法暫停它。

那么,這兩種模式有什么區(qū)別呢?在我理解,如果你不需要對(duì)數(shù)據(jù)進(jìn)行精確控制,首先選擇流動(dòng)模式,因?yàn)樗男矢?。如果需要?duì)流的過程進(jìn)行精確控制則可以選擇暫停模式。也就是說暫停模式是流更高級(jí)一些的用法。其實(shí)官方建議我們盡量不要手動(dòng)去操作流,如果可以,盡量使用pipe方法。

接下來,不論我們以哪種方式讀到了文件中的數(shù)據(jù),這時(shí)我們都可以創(chuàng)建一個(gè)可寫流并調(diào)用可寫流的write方法來消費(fèi)讀到的數(shù)據(jù)。調(diào)用write方法會(huì)向文件中寫入數(shù)據(jù),但是因?yàn)閷懭氲乃俣容^慢,如果當(dāng)前寫入還在進(jìn)行,而你又調(diào)用了write方法,node會(huì)將你要寫入的數(shù)據(jù)緩存在一個(gè)緩存區(qū)中,等到文件寫入完畢會(huì)從緩存區(qū)中取出數(shù)據(jù),繼續(xù)寫入。

write方法擁有一個(gè)布爾類型的返回值,用來表示目前是否還可以繼續(xù)調(diào)用write方法寫入內(nèi)容。如果返回false,我們應(yīng)當(dāng)停止讀取數(shù)據(jù)以避免消耗過多內(nèi)存。那么什么時(shí)候會(huì)返false呢?就是當(dāng)緩存區(qū)的大小大于16k(可以在創(chuàng)建可讀流時(shí),通過option參數(shù)中的highWaterMark更改)時(shí)。

緩存區(qū)滿后,文件寫入一直在進(jìn)行,不一會(huì)兒會(huì)把緩存區(qū)的內(nèi)容全部寫入,緩存區(qū)處于清空狀態(tài),這時(shí)會(huì)觸發(fā)可寫流的‘drain'事件,這時(shí)我們可以繼續(xù)向文件寫入數(shù)據(jù)了。注意:如果緩存區(qū)從未滿過,‘drain'事件永遠(yuǎn)也不會(huì)觸發(fā)。

那么這張圖對(duì)應(yīng)到代碼是什么樣的呢:

let fs = require('fs');
//創(chuàng)建可讀可寫流
let rs = fs.createReadStream('./1.txt');
let ws = fs.createWriteStream('./2.txt');
//監(jiān)聽‘data'事件,開啟流動(dòng)模式
rs.on('data',function (data) {
  //對(duì)應(yīng)圖中的可寫流true和false
  let flag = ws.write(data);
  if(!flag){
    //如果可寫流返回false,我們應(yīng)當(dāng)停止讀取,以避免消耗過多內(nèi)存
    rs.pause();
  }
});

//對(duì)應(yīng)圖中的drain事件
ws.on('drain',function () {
  //重新開啟流動(dòng)模式
  rs.resume();
});

//使用可讀流的暫停模式
function read() {
  let data = rs.read()
  let flag = ws.write(data);
  if(flag){
   read()
  }
}

rs.on('readable',function(){
  read()
})

ws.on('drain',function () {
  read()
});

結(jié)尾

可讀和可寫流的用法和api還有很多,這里只是簡單的梳理了一下基本過程,如果有描述不準(zhǔn)確的地方還請大家在評(píng)論區(qū)多指正。

參考資料

- https://nodejs.org/dist/latest-v8.x/docs/api/stream.html

- http://nodejs.cn/api/

以上就是本文的全部內(nèi)容,希望對(duì)大家的學(xué)習(xí)有所幫助,也希望大家多多支持創(chuàng)新互聯(lián)。

網(wǎng)頁名稱:說說node中的可讀流和可寫流的區(qū)別
URL網(wǎng)址:http://muchs.cn/article30/ispopo.html

成都網(wǎng)站建設(shè)公司_創(chuàng)新互聯(lián),為您提供網(wǎng)站設(shè)計(jì)公司、網(wǎng)站內(nèi)鏈、外貿(mào)網(wǎng)站建設(shè)做網(wǎng)站、品牌網(wǎng)站制作、搜索引擎優(yōu)化

廣告

聲明:本網(wǎng)站發(fā)布的內(nèi)容(圖片、視頻和文字)以用戶投稿、用戶轉(zhuǎn)載內(nèi)容為主,如果涉及侵權(quán)請盡快告知,我們將會(huì)在第一時(shí)間刪除。文章觀點(diǎn)不代表本網(wǎng)站立場,如需處理請聯(lián)系客服。電話:028-86922220;郵箱:631063699@qq.com。內(nèi)容未經(jīng)允許不得轉(zhuǎn)載,或轉(zhuǎn)載時(shí)需注明來源: 創(chuàng)新互聯(lián)

成都app開發(fā)公司