亚洲乱码中文字幕综合,中国熟女仑乱hd,亚洲精品乱拍国产一区二区三区,一本大道卡一卡二卡三乱码全集资源,又粗又黄又硬又爽的免费视频

深入淺出了解Node.js Streams

 更新時間:2019年05月27日 14:50:37   作者:Yzz  
這篇文章講了了解流的用途,為什么它們?nèi)绱酥匾?,以及如何使用它們。下面我們來一起學習吧

什么是流(steams)

流(stream)是 Node.js 中處理流式數(shù)據(jù)的抽象接口。

Streams 不是 Node.js 獨有的概念。它們是幾十年前在 Unix 操作系統(tǒng)中引入的。

它們能夠以一種有效的方式來處理文件的讀、寫,網(wǎng)絡通信或任何類型的端到端信息交換。
例如,當你編寫了一段程序用來讀取文件時,傳統(tǒng)的方法是將文件從頭到尾讀入內(nèi)存,然后再進行處理。而使用流的話,你就可以逐塊讀取它,處理其內(nèi)容而不將其全部保存在內(nèi)存中。
以如下代碼為例

const fs = require('fs');
const rs = fs.createReadStream('test.md');
let data = '';
rs.on("data", function (chunk) {
data += chunk;
});
rs.on("end", function() {
console.log(data);
});

利用 createReadStream 創(chuàng)建一個讀取數(shù)據(jù)的流,來讀取 test.md 文件的內(nèi)容,此時監(jiān)聽 data 事件,它是在當流將數(shù)據(jù)塊傳送給消費者后觸發(fā)。并在對應的 eventHandler 中,拼接 chunk。在 end 事件中,打印到終端上。
之前說流,可以逐塊讀取文件內(nèi)容,那么這個塊,也就是 chunk 是什么?
一般情況下是 Buffer,修改 data 事件的 eventHandler 來驗證下

rs.on("data", function (chunk) {
console.log("chunk", Buffer.isBuffer(chunk)) // log true
data += chunk;
});

流的工作方式可以具體的表述為,在內(nèi)存中準備一段 Buffer,然后在 fs.read() 讀取時逐步從磁盤中將字節(jié)復制到 Buffer 中。

為什么要使用 Stream

利用 Stream 來處理數(shù)據(jù),主要是因為它的兩個優(yōu)點:

內(nèi)存效率:在夠處理數(shù)據(jù)之前,不需要占用大量內(nèi)存;

時間效率:處理數(shù)據(jù)花費的時間更少,因為流是逐塊來處理數(shù)據(jù),而不是等到整個數(shù)據(jù)有效負載才啟動。

首先內(nèi)存效率,與 fs.readFile 這種會緩沖整個文件相比,流式傳輸充分地利用 Buffer (超過 8kb)不受 V8 內(nèi)存控制的特點,利用堆外內(nèi)存完成高效地傳輸。相關(guān)驗證可以參考這篇博文,地址。
時間效率,與 fs.FileSync 相比,有些優(yōu)勢,但是與異步的 fs.readFile 相比,優(yōu)勢不大。

Node.js 中 Stream 的使用

首先用一張圖來了解下 Node.js 中有哪些內(nèi)置的 Stream 接口

圖中提供了一些 Node.js 原生的流的示例,有些是可讀、寫的流。 也有一些是可讀寫的流,如 TCP sockets、zlib 以及 crypto。

特別注意: 流的讀、寫與環(huán)境是密切相關(guān)的。例如 HTTP 響應在客戶端上的可讀流,但它是服務器上的可寫流。同時還需要注意,stdio streams(stdin,stdout,stderr) 在子進程上是相反的流。

使用一個例子來展示流的使用

首先利用如下腳本創(chuàng)建一個比較大的文件(大概 430 MB)

const fs = require('fs');
const file = fs.createWriteStream('test.md');
for(let i=0; i<= 1e6; i++) {
file.write('hello world.\n');
}
file.end();

在當前目錄下,啟動 http 服務

const http = require('http')
const fs = require('fs')
const server = http.createServer(function (req, res) {
fs.readFile(__dirname + '/test.md', (err, data) => {
res.end(data)
})
})
server.listen(3000)

得到的結(jié)果,如圖

const http = require('http')
const fs = require('fs')
const server = http.createServer((req, res) => {
const stream = fs.createReadStream(__dirname + '/test.md')
stream.pipe(res)
})
server.listen(3000)

時間減少了 2s 多。這可以解釋為,在讀取文件內(nèi)容,并且不需要改變內(nèi)容的場景下,流能夠完成只讀取 buffer,然后直接傳輸,不做額外的轉(zhuǎn)換,避免損耗,提高性能。
上述代碼中,應用了 stream.pipe(...) 。它主要是對流進行鏈式地管道操作,例如

src.pipe(dest1).pipe(dest2)

這樣數(shù)據(jù)流會被自動管理。

如果可讀流發(fā)生錯誤,目標可寫流不會自動關(guān)閉,需要手動關(guān)閉所有流以避免內(nèi)存泄漏。

通常,當你使用 pipe 方法時,就不需要使用事件,但如果場景需要以更靈活、自定義的方式使用流,那么就要考慮事件。

Stream events

在上述例子中,我們使用了可讀流的 data 、end 事件來控制文件的讀取,它本質(zhì)上與 pipe 方法相同,例如

# readable.pipe(writable)
readable.on('data', (chunk) => {
writable.write(chunk);
});
readable.on('end', () => {
writable.end();
});

只不過,使用 event 會更加靈活,可控。

圖中簡單羅列了可讀流、可寫流的相關(guān)事件、方法,其中最重要的是

可讀流:

  • data 事件:每當流將一大塊數(shù)據(jù)傳遞時,就會觸發(fā);
  • end 事件:當沒有更多數(shù)據(jù)要從流發(fā)出時,就會觸發(fā)。

可寫流:

  • drain 事件:當可以繼續(xù)寫入數(shù)據(jù)到流時會觸發(fā)事件;
  • finish 事件:處理完全部數(shù)據(jù)塊之后觸發(fā)。

流的不同類型

除了上面涉及到的可讀、寫流之后,還有 Duplex、Transform 兩類:

  • Readable :可以接收數(shù)據(jù),但不能向其發(fā)送數(shù)據(jù)。當你將數(shù)據(jù)推送到可讀流中時,它會被緩沖,直到消費者開始讀取數(shù)據(jù);
  • writable :可以發(fā)送數(shù)據(jù),但不能從中接收;
  • Duplex :即可讀也可寫;
  • Tranform :與 Duplex 一樣是可寫又可讀的,但它的輸出與輸入是相關(guān)聯(lián)的。

如何創(chuàng)建一個可讀流

這里只做簡單介紹,具體見 stream module。

const Stream = require('stream')
const readableStream = new Stream.Readable()
readableStream._read = (size) => {
console.log('read', size)
}

利用 Stream 模塊初始化一個可讀流,然后向其中發(fā)送數(shù)據(jù)

readableStream.push('hi!')
readableStream.push('ho!')

如何創(chuàng)建一個可寫流

為了創(chuàng)建可寫流,需要擴展了基本的 Writable 對象,并實現(xiàn)了它的 _write 方法。

const Stream = require('stream')
const writableStream = new Stream.Writable()

實現(xiàn) _write 方法:

writableStream._write = (chunk, encoding, next) => {
console.log(chunk.toString())
next()
}

結(jié)合上述例子實現(xiàn)

利用 readableStream 讀入數(shù)據(jù),并輸出到 writableStream

const Stream = require('stream')
const readableStream = new Stream.Readable()
readableStream._read = (size) => {
console.log('read', size)
}
const writableStream = new Stream.Writable()
writableStream._write = (chunk, encoding, next) => {
console.log('write', chunk.toString())
next()
}
readableStream.pipe(writableStream)
readableStream.push('hi!')
readableStream.push('ho!')
/* 
log:
read 16384
write hi!
write ho!
*/

以上就是本文的全部內(nèi)容,希望對大家的學習有所幫助,也希望大家多多支持腳本之家。

相關(guān)文章

  • nodejs socket實現(xiàn)的服務端和客戶端功能示例

    nodejs socket實現(xiàn)的服務端和客戶端功能示例

    這篇文章主要介紹了nodejs socket實現(xiàn)的服務端和客戶端功能,結(jié)合具體實例形式分析了nodejs基于socket通信實現(xiàn)的服務端與客戶端功能相關(guān)操作技巧,需要的朋友可以參考下
    2017-06-06
  • node.js文件上傳重命名以及移動位置的示例代碼

    node.js文件上傳重命名以及移動位置的示例代碼

    本篇文章主要介紹了node.js文件上傳重命名以及移動位置的示例代碼,小編覺得挺不錯的,現(xiàn)在分享給大家,也給大家做個參考。一起跟隨小編過來看看吧
    2018-01-01
  • 使用nodejs?+?koa?+?typescript?集成和自動重啟的問題

    使用nodejs?+?koa?+?typescript?集成和自動重啟的問題

    這篇文章主要介紹了nodejs?+?koa?+?typescript?集成和自動重啟,本文給大家介紹的非常詳細,對大家的學習或工作具有一定的參考借鑒價值,需要的朋友可以參考下
    2021-12-12
  • 通過node-mysql搭建Windows+Node.js+MySQL環(huán)境的教程

    通過node-mysql搭建Windows+Node.js+MySQL環(huán)境的教程

    這篇文章主要介紹了通過node-mysql搭建Windows+Node.js+MySQL環(huán)境的教程,node-mysql是JavaScript編寫的一個Node的MySQL驅(qū)動,需要的朋友可以參考下
    2016-03-03
  • 使用nodejs搭建一個簡易HTTP服務的實現(xiàn)示例

    使用nodejs搭建一個簡易HTTP服務的實現(xiàn)示例

    本文主要介紹了使用nodejs搭建一個簡易HTTP服務的實現(xiàn)示例,文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友們下面隨著小編來一起學習學習吧
    2022-05-05
  • 淺談手寫node可讀流之流動模式

    淺談手寫node可讀流之流動模式

    這篇文章主要介紹了淺談手寫node可讀流之流動模式,小編覺得挺不錯的,現(xiàn)在分享給大家,也給大家做個參考。一起跟隨小編過來看看吧
    2018-06-06
  • node.js遍歷目錄的方法示例

    node.js遍歷目錄的方法示例

    本篇文章主要介紹了node.js遍歷目錄的方法示例,主要介紹了同步遍歷和異步遍歷兩種方法,非常具有實用價值,需要的朋友可以參考下
    2018-08-08
  • 切換Node.js鏡像源的方法詳解

    切換Node.js鏡像源的方法詳解

    Node.js 是一個非常流行的 JavaScript 運行環(huán)境,擁有許多強大的功能和庫,使得它在 Web 開發(fā)、服務器端編程等領(lǐng)域中廣泛使用,在本文中,我們將介紹如何切換 Node.js 的鏡像源,并提供一些思考和難點,幫助讀者更好地理解這個問題,需要的朋友可以參考下
    2024-01-01
  • Nodejs使用express連接數(shù)據(jù)庫mongoose的示例

    Nodejs使用express連接數(shù)據(jù)庫mongoose的示例

    數(shù)據(jù)庫并進行操作通常需要使用第三方庫,其中最流行的是mongoose,本文主要介紹了Nodejs使用express連接數(shù)據(jù)庫mongoose的示例,具有一定的參考價值,感興趣的可以了解一下
    2024-06-06
  • Nuxt配合Node在實際生產(chǎn)中的應用詳解

    Nuxt配合Node在實際生產(chǎn)中的應用詳解

    這篇文章主要介紹了Nuxt配合Node在實際生產(chǎn)中的應用詳解,小編覺得挺不錯的,現(xiàn)在分享給大家,也給大家做個參考。一起跟隨小編過來看看吧
    2018-08-08

最新評論