欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

Nodejs 數(shù)組的隊(duì)列以及forEach的應(yīng)用詳解

 更新時間:2021年02月25日 16:01:26   作者:WificamSDK7  
這篇文章主要介紹了Nodejs 數(shù)組的隊(duì)列以及forEach的應(yīng)用詳解,具有很好的參考價值,希望對大家有所幫助。一起跟隨小編過來看看吧

本文主要記錄了在Nodejs開發(fā)過程中遇到過的由數(shù)組特性引起的問題及解決方式,以及對數(shù)組的靈活應(yīng)用。

本文代碼測試結(jié)果均基于node v6.9.5

數(shù)組與隊(duì)列

利用數(shù)組對象方法push/shift可實(shí)現(xiàn)隊(duì)列先進(jìn)先出特性,例如:

>a=[]
[]
>a.push(2.3.4)
3
>a.push(2)
3
>a
[2.3.4.2]
>a.shift()
2
>a
>[3.4.2]

數(shù)組與forEach

對數(shù)組的刪除操作有兩種常見方式:delete和使用splice方法,需要明確他們的區(qū)別。

操作/方法 說明
splice 刪除并返回指定的數(shù)組元素,數(shù)組本身長度會改變;但不會free元素對象
delete 刪除(free)元素對象,數(shù)組元素不變,值變?yōu)閡ndefined

如果要從數(shù)組中徹底刪除某個元素,使用splice即可:

> a=[1,2,3]
[ 1, 2, 3 ]
> a.splice(1,1)
[ 2 ]
> a
[ 1, 3 ]
> a.length
2
> a.forEach(function(item, index){console.info("index[", index,"]:", item)});
index[ 0 ]: 1
index[ 1 ]: 3
undefined
> 

那么,當(dāng)使用delete刪除某個元素對象后,此時執(zhí)行forEach的效果是什么?

forEach對含空元素數(shù)組處理機(jī)制

測試結(jié)果如下

> a=[1,2,3]
[ 1, 2, 3 ]
> delete a[1]
true
> a
[ 1, , 3 ]
> a.length
3
> a.forEach(function(item, index){console.info("index[", index,"]:", item)});
index[ 0 ]: 1
index[ 2 ]: 3
undefined

從測試結(jié)果來看,forEach并不會遍歷到值為undefined的哪一項(xiàng)。這在實(shí)際應(yīng)用中如何判斷forEach是否結(jié)束是一大挑戰(zhàn)。

解決配合forEach的異步特性應(yīng)用,可為數(shù)組添加prototype來自行管理設(shè)置有效數(shù)據(jù);

效果如下:

> a=[1,2,3]
[ 1, 2, 3 ]
> a.validnum=3
3
> delete a[2]
true
> a.validnum=2
2
> a
[ 1, 2, , validnum: 2 ]
> a.length
3
> a.validnum
2
> a.forEach(function(item, index){console.info("index[", index,"]:", item)});
index[ 0 ]: 1
index[ 1 ]: 2
undefined
> 

補(bǔ)充:Node.js 數(shù)組 forEach 同步處理上下文語句

習(xí)慣了C語言系的思維方式,剛接觸Node.js,它的異步處理讓我頭大。

寫代碼遇到這么一個場景,需要循環(huán)對一個數(shù)組中的元素進(jìn)行處理,全部處理完成后再執(zhí)行一個last操作。但是JS的異步特性會使這個last語句先執(zhí)行,所以花點(diǎn)時間研究研究forEach。

Talk is cheap. Show me the code.

forEach 用法

forEach用于對數(shù)組結(jié)構(gòu)進(jìn)行遍歷,看到有人說forEach底層是用for實(shí)現(xiàn)的,沒深究,起碼效果上看是一樣的。forEach的回調(diào)函數(shù)3個參數(shù)分別是:值、序號和原數(shù)組。序號從0開始。

(() => {
  let arr = [2, 3, 1];
  arr.forEach(function (value, index, array) {
    console.log(value);
    console.log(index);
    console.log(array);
    console.log('-----');
  });
})();

Output

2
0
[ 2, 3, 1 ]
-----
3
1
[ 2, 3, 1 ]
-----
1
2
[ 2, 3, 1 ]
-----

從結(jié)果上看forEach多次循環(huán)之間是同步的,也就是說都是按順序執(zhí)行的。但是一想到它是JS就感覺不可能同步的。。可以驗(yàn)證一下。

forEach 異步處理多次循環(huán)

這次在forEach加個定時任務(wù),每次循環(huán)操作都延時value相關(guān)的時間,模擬比較耗時的操作。

(() => {
  let arr = [2, 3, 1];
  arr.forEach(function (value, index, array) {
    setTimeout(function () {
      console.log(value);
    }, value*100);
  });
})();

Output

1
2
3

從結(jié)果可以看出耗時最短的任務(wù)先完成,每次循環(huán)的任務(wù)并不是按循環(huán)的先后順序執(zhí)行的,也就是說異步處理多次循環(huán)。

forEach 上下文也是異步執(zhí)行

回到開始說到的問題了,且不管多次循環(huán)是不是按順序執(zhí)行,我需要forEach中的所有任務(wù)都完成后執(zhí)行一條數(shù)據(jù)來通知我任務(wù)全部完成了。

(() => {
  let arr = [2, 3, 1];
  arr.forEach(function (value, index, array) {
    setTimeout(function () {
      console.log(value);
    }, value*100);
  });
  console.log('All the work is done');
})();

Output

All the work is done
1
2
3

從結(jié)果來看,上下文的語句也不是同步的,forEach循環(huán)中的任務(wù)沒有完成就通知所有任務(wù)都完成了,顯然不符合預(yù)期。

針對這個問題看了好多個博客,都沒有找到合適的解決方法,最后只能想到用Promise.all來勉強(qiáng)實(shí)現(xiàn)這個功能。

Promise.all 實(shí)現(xiàn) forEach 上下文語句同步處理

把上面的代碼改成Promise.all的結(jié)構(gòu)。每個循環(huán)中執(zhí)行結(jié)束調(diào)用resolve(),我們知道Promise.all的then函數(shù),只有所有的Promise都執(zhí)行完成才會觸發(fā),這樣好像能滿足我們的需求。

(() => {
  let arr = [2, 3, 1];
  let proArr = [];
  arr.forEach(function (value, index) {
    proArr[index] = new Promise(function (resolve) {
      setTimeout(function () {
        console.log(value);
        resolve();
      }, value*100);
    });
  });
  Promise.all(proArr).then(()=>{
    console.log('All the work is done');
  })
})();

Output

1
2
3
All the work is done

從結(jié)果來看,滿足了我們的需求。

可能還存在的問題

想到JS異步特性,突然發(fā)現(xiàn)可能這個方法還存在個問題。

這里每次 forEach 剛進(jìn)入就對 Promise 數(shù)組進(jìn)行了賦值操作,這個操作時間應(yīng)該非常短,循環(huán)3次都賦值完成后才調(diào)用最后的Promise.all語句。

但是如果這個數(shù)組非常大,這個循環(huán)賦值的操作非常耗時間的話,假如只完成了一半的賦值操作,那么執(zhí)行最后這個 Promise.all 的時候傳入的 Promise 數(shù)組可能并不是包含所有 Promise 的數(shù)組。

這樣的話 Promise.all 等待的就只有一半的操作,Promise.all 等待的時候,這個數(shù)組后面被賦值的 Promise 不知道會不會被等待。

剛接觸JS不明白實(shí)現(xiàn)機(jī)制,只能實(shí)驗(yàn)來驗(yàn)證一下是否存在這個問題。接下來用把這個數(shù)組弄大一些,請原諒我用最傻瓜式的方式搞大它。

(() => {
  let arr = [2, 3, 1, 2, 3, 1, 2, 3, 1, 2];  // 10
  arr= arr.concat(arr);  // 2^1 * 10
  arr= arr.concat(arr);  // 2^2 * 10
  arr= arr.concat(arr);  // 2^3
  arr= arr.concat(arr);  // 2^4
  arr= arr.concat(arr);  // 2^5
  arr= arr.concat(arr);
  arr= arr.concat(arr);
  arr= arr.concat(arr);
  arr= arr.concat(arr);
  arr= arr.concat(arr);  // 2^10
  arr= arr.concat(arr);
  arr= arr.concat(arr);
  arr= arr.concat(arr);
  arr= arr.concat(arr);
  arr= arr.concat(arr);  // 2^15
  arr= arr.concat(arr);
  arr= arr.concat(arr); // 2^17 * 10
// arr= arr.concat(arr);  // 2^18 * 10
  console.log(arr.length);
  let proArr = [];
  arr.forEach(function (value, index) {
    proArr[index] = new Promise(function (resolve) {
      setTimeout(function () {
        console.log(value);
        resolve();
      }, value*100);
    });
  });
  Promise.all(proArr).then(()=>{
    console.log('All the work is done');
    console.log(arr.length);
  }).catch(function (err) {
    console.log(err);
  })
})();

經(jīng)過測試在我這個電腦上當(dāng)數(shù)組長度為2^18 * 10的時候,Promise報錯 RangeError: Too many elements passed to Promise.all。

當(dāng)數(shù)組長度為2^17 * 10 即2621440的時候,會正常運(yùn)行。測試了幾次,最后的執(zhí)行命令輸出的All the work is done始終在最后輸出(因?yàn)榻K端緩沖區(qū)太小,所以使用node xx.js > log.txt重定向的方式把輸出結(jié)果重定向到文件查看)。

當(dāng)然應(yīng)用中也不會有這么大的數(shù)組,從結(jié)果看的話,就是實(shí)際應(yīng)用中不存在上面考慮可能出現(xiàn)的問題。

也就是說可以用 Promise.all 實(shí)現(xiàn) forEach 上下文語句同步處理。

以上為個人經(jīng)驗(yàn),希望能給大家一個參考,也希望大家多多支持腳本之家。如有錯誤或未考慮完全的地方,望不吝賜教。

相關(guān)文章

最新評論