4 個(gè)JavaScript 中 array.reduce() 數(shù)組方法的使用實(shí)例

英文 | https://javascript.plainenglish.io/4-practices-to-help-you-understand-array-reduce-f3138cfef095
翻譯 | 楊小愛(ài)
Array.prototype.reduce() 是數(shù)組中最強(qiáng)大的方法之一,也是 JavaScript 函數(shù)式編程中一個(gè)吸引人的特性。但不幸的是,我發(fā)現(xiàn)很多朋友不習(xí)慣使用它。
今天請(qǐng)讓我詳細(xì)介紹一下這個(gè)方法,希望對(duì)你有幫助。
這是 reduce 的基本用法:
var arr = [1, 2, 3];function reducer(parmar1, parmar2){}arr.reduce(reducer)
reduce是數(shù)組原型對(duì)象上的一個(gè)方法,可以幫助我們操作數(shù)組。它將另一個(gè)函數(shù)作為其參數(shù),可以稱為reducer。
reducer 有兩個(gè)參數(shù)。第一個(gè)參數(shù) param1 是最后一次 reducer 運(yùn)行的結(jié)果。如果這是第一次運(yùn)行 reducer,則 param1 的默認(rèn)值是數(shù)組第一個(gè)元素的值。
reduce 方法循環(huán)遍歷數(shù)組中的每個(gè)元素,就像在 for 循環(huán)中一樣。并將循環(huán)中的當(dāng)前值作為參數(shù)2。
遍歷完數(shù)組,reduce會(huì)返回最后一個(gè)reducer計(jì)算的結(jié)果。
我們來(lái)看一個(gè)詳細(xì)的例子。
var arr = ['a', 'b', 'c', 'd', 'e'];function add(x, y) {return x + y;}arr.reduce(add)

接下來(lái),我們來(lái)探索一下上面的代碼是如何執(zhí)行的。
在這段代碼中,reducer 是 add 。
首先,因?yàn)槲覀兪堑谝淮螆?zhí)行add,所以數(shù)組中的第一個(gè)元素'a'會(huì)被當(dāng)作add的第一個(gè)參數(shù),然后循環(huán)會(huì)從數(shù)組的第二個(gè)元素'b'開(kāi)始。這一次,'b' 是 add 的第二個(gè)參數(shù)。

第一次計(jì)算后,我們得到結(jié)果'ab'。此結(jié)果將被緩存并在下一次添加計(jì)算中用作 param1。同時(shí),數(shù)組中的第三個(gè)參數(shù)'c'將用作add的param2。

同樣,reduce 會(huì)繼續(xù)遍歷數(shù)組中的元素,運(yùn)行 'abc' 和 'd' 作為 add 的參數(shù)。

最后,遍歷數(shù)組中最后一個(gè)元素后,返回計(jì)算結(jié)果。

現(xiàn)在我們有了結(jié)果:'abcde'。
所以,我們可以看到reduce也是一種遍歷數(shù)組的方式!它依次取數(shù)組中每個(gè)元素的值并執(zhí)行reducer函數(shù)。
但我們可以看到,上面的循環(huán)并沒(méi)有那種和諧的美感。因?yàn)槲覀儼褦?shù)組的第一個(gè)元素,也就是'a'作為初始的param1,然后從數(shù)組的第二個(gè)元素循環(huán)得到param2。
實(shí)際上,我們可以將reduce中的第二個(gè)參數(shù)指定為reducer函數(shù)的param1的初始值,這樣param2就會(huì)從數(shù)組的第一個(gè)元素開(kāi)始循環(huán)獲取。
代碼如下:
var arr = ['a', 'b', 'c', 'd', 'e'];function add(x, y) {return x + y;}arr.reduce(add, 's')

這一次,我們第一次調(diào)用reducer時(shí)將's'作為param1,然后從第一個(gè)元素開(kāi)始依次遍歷數(shù)組。

所以我們可以使用這個(gè)語(yǔ)法來(lái)重寫(xiě)我們的第一個(gè)代碼片段。
var arr = ['a', 'b', 'c', 'd', 'e'];function add(x, y) {return x + y;}arr.reduce(add, '')
接下來(lái),我們將進(jìn)入實(shí)際編程章節(jié),體驗(yàn)reduce的強(qiáng)大威力。
1、累加和累積乘法
如果我們想得到數(shù)組中所有元素的總和,你會(huì)怎么做?
一般來(lái)說(shuō),你可能會(huì)這樣寫(xiě):
function accumulation(arr) {let sum = 0;for (let i = 0; i < arr.length; i++) {sum = sum + arr[i];}return sum;}
當(dāng)然,你可能還有其他的寫(xiě)法,但是只要使用for循環(huán),代碼就會(huì)顯得多余。
那我們看看上面的累加函數(shù)是做什么的:
將初始總和設(shè)置為零
取出數(shù)組中的第一個(gè)元素并求和
在 sum 中緩存上一步的結(jié)果
依次取出數(shù)組中的其他元素,進(jìn)行上述操作
返回最終結(jié)果
我們可以看到,當(dāng)我們用文字描述上述步驟時(shí),很明顯它符合reduce的使用。所以我們可以使用reduce來(lái)重寫(xiě)上面的代碼:
function accumulation(arr) {function reducer(x, y) {return x + y}return arr.reduce(reducer, 0);}
如果你習(xí)慣使用箭頭函數(shù),上面的代碼看起來(lái)會(huì)更簡(jiǎn)潔:
function accumulation(arr) {return arr.reduce((x, y) => x + y, 0);}
一行代碼搞定!

當(dāng)然,累積乘法和累加是完全一樣的:
function multiplication(arr) {return arr.reduce((x, y) => x * y, 1);}
很多時(shí)候,我們?cè)谇蠛偷臅r(shí)候需要加上一個(gè)權(quán)重,這樣更能體現(xiàn)reduce的優(yōu)雅。
const scores = [{ score: 90, subject: "HTML", weight: 0.2 },{ score: 95, subject: "CSS", weight: 0.3 },{ score: 85, subject: "JavaScript", weight: 0.5 }];const result = scores.reduce((x, y) => x + y.score * y.weight, 0); // 89
2、獲取一個(gè)數(shù)組的最大值和最小值
如果要獲取數(shù)組的最大值和最小值,可以這樣寫(xiě):
function max(arr){let max = arr[0];for (let ele of arr) {if(ele > max) {max = ele;}}return max;}
這和以前一樣,如果我們使用reduce,我們可以在一行代碼中完成。
let arr = [3.24, 2.78, 999];arr.reduce((x, y) => Math.max(x, y));arr.reduce((x, y) => Math.min(x, y));

3、計(jì)算數(shù)組中元素出現(xiàn)的頻率
我們經(jīng)常需要統(tǒng)計(jì)數(shù)組中每個(gè)元素出現(xiàn)的次數(shù)。reduce 方法可以幫助我們實(shí)現(xiàn)這一點(diǎn)。
function countFrequency(arr) {return arr.reduce(function(result, ele){// Judge whether this element has been counted beforeif (result.get(ele) != undefined) {/*** If this element has been counted before,* increase the frequency of its occurrence by 1*/result.set(ele, result.get(ele) + 1)} else {/*** If this element has not been counted before,* set the frequency of its occurrence to 1*/result.set(ele, 1);}return result;}, new Map());}
注意,我們使用map對(duì)象而不是對(duì)象來(lái)存儲(chǔ)統(tǒng)計(jì)后的頻率,因?yàn)閿?shù)組中的元素可能是對(duì)象類(lèi)型,而對(duì)象的key只能是字符串或符號(hào)類(lèi)型。
這里有兩個(gè)例子:


同樣,如果要統(tǒng)計(jì)字符串中每個(gè)字符出現(xiàn)的頻率,可以先將字符串轉(zhuǎn)換為字符數(shù)組,然后按照上面的方法。
let str = 'helloworld';str.split('').reduce((result, currentChar) => {result[currentChar] ? result[currentChar] ++ : result[currentChar] = 1;return result;}, {})

因?yàn)樽址?lèi)型可以用作對(duì)象的鍵,所以我們這里不使用 Map。
4、多個(gè)數(shù)組的展平
function Flat(arr = []) {return arr.reduce((t, v) => t.concat(Array.isArray(v) ? Flat(v) : v), [])}

通過(guò)reduce依次訪問(wèn)數(shù)組中的每個(gè)元素。如果我們發(fā)現(xiàn)元素還是一個(gè)數(shù)組,就遞歸調(diào)用 flat 方法。
總結(jié)
以上就是我今天跟你分享的4個(gè)關(guān)于數(shù)組reduce的用法,希望對(duì)你有所幫助。
學(xué)習(xí)更多技能
請(qǐng)點(diǎn)擊下方公眾號(hào)
![]()

