国产探花免费观看_亚洲丰满少妇自慰呻吟_97日韩有码在线_资源在线日韩欧美_一区二区精品毛片,辰东完美世界有声小说,欢乐颂第一季,yy玄幻小说排行榜完本

首頁 > 編程 > JavaScript > 正文

用NODE.JS中的流編寫工具是要注意的事項

2019-11-20 10:28:34
字體:
來源:轉載
供稿:網友

Node.js中的流十分強大,它對處理潛在的大文件提供了支持,也抽象了一些場景下的數據處理和傳遞。正因為它如此好用,所以在實戰中我們常?;谒鼇砭帉懸恍┕ぞ?函數/庫 ,但往往又由于自己對流的某些特性的疏忽,導致寫出的 函數/庫 在一些情況會達不到想要的效果,或者埋下一些隱藏的地雷。本文將會提供兩個在編寫基于流的工具時,私以為有些用的兩個tips。

一,警惕EVENTEMITTER內存泄露

在一個可能被多次調用的函數中,如果需要給流添加事件監聽器來執行某些操作。那么則需要警惕添加監聽器而導致的內存泄露:

'use strict';const fs = require('fs');const co = require('co');function getSomeDataFromStream (stream) { let data = stream.read(); if (data) return Promise.resolve(data); if (!stream.readable) return Promise.resolve(null); return new Promise((resolve, reject) => {  stream.once('readable', () => resolve(stream.read()));  stream.on('error', reject);  stream.on('end', resolve); })}let stream = fs.createReadStream('/Path/to/a/big/file');co(function *() { let chunk; while ((chunk = yield getSomeDataFromStream(stream)) !== null) {  console.log(chunk); }}).catch(console.error);

在上述代碼中,getSomeDataFromStream函數會在通過監聽error事件和end事件,來在流報錯或沒有數據時,完成這個Promise。然而在執行代碼時,我們很快就會在控制臺中看到報警信息:(node) warning: possible EventEmitter memory leak detected. 11 error listeners added. Use emitter.setMaxListeners() to increase limit.,因為我們在每次調用該函數時,都為傳入的流添加了一個額外的error事件監聽器和end事件監聽器。為了避免這種潛在的內存泄露,我們要確保每次函數執行完畢后,清除所有此次調用添加的額外監聽器,保持函數無污染:

function getSomeDataFromStream (stream) { let data = stream.read(); if (data) return Promise.resolve(data); if (!stream.readable) return Promise.resolve(null); return new Promise((resolve, reject) => {  stream.once('readable', onData);  stream.on('error', onError);  stream.on('end', done);  function onData () {   done();   resolve(stream.read());  }  function onError (err) {   done();   reject(err);  }  function done () {   stream.removeListener('readable', onData);   stream.removeListener('error', onError);   stream.removeListener('end', done);  } })}

二,保證工具函數的回調在處理完畢數據后才被調用

工具函數往往會對外提供一個回調函數參數,待處理完流中的所有數據后,帶著指定值觸發,通常的做法是將回調函數的調用掛在流的end事件中,但如果處理函數是耗時的異步操作,回調函數則可能在所有數據處理完畢前被調用:

'use strict';const fs = require('fs');let stream = fs.createReadStream('/Path/to/a/big/file');function processSomeData (stream, callback) { stream.on('data', (data) => {  // 對數據進行一些異步耗時操作  setTimeout(() => console.log(data), 2000); }); stream.on('end', () => {  // ...  callback() })}processSomeData(stream, () => console.log('end'));

以上的代碼callback回調可能會在數據并未被全部處理時就被調用,因為流的end事件的觸發時機僅僅是在流中的數據被讀完時。所以我們需要額外地對數據是否已處理完進行檢查:

function processSomeData (stream, callback) { let count = 0; let finished = 0; let isEnd = false; stream.on('data', (data) => {  count++;  // 對數據進行一些異步耗時操作  setTimeout(() => {   console.log(data);   finished++;   check();  }, 2000); }); stream.on('end', () => {  isEnd = true;  // ...  check(); }) function check () {  if (count === finished && isEnd) callback() }}

這樣一來,回調便會在所有數據都處理完畢后觸發了。

發表評論 共有條評論
用戶名: 密碼:
驗證碼: 匿名發表
主站蜘蛛池模板: 桐乡市| 林口县| 茶陵县| 和静县| 渭源县| 金昌市| 象山县| 连南| 新沂市| 永平县| 黎城县| 金门县| 兴国县| 信宜市| 台北县| 台南县| 崇义县| 鄂伦春自治旗| 锦屏县| 柯坪县| 南岸区| 金门县| 辉南县| 龙里县| 嫩江县| 宜川县| 汪清县| 上思县| 武功县| 甘孜县| 盖州市| 黄冈市| 文化| 元江| 宁陕县| 罗田县| 阳东县| 启东市| 金寨县| 屯留县| 四平市|