JavaScript中array.reduce()數(shù)組方法的四種使用實(shí)例
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é)果。
我們來看一個(gè)詳細(xì)的例子。
var arr = ['a', 'b', 'c', 'd', 'e']; function add(x, y) { return x + y; } arr.reduce(add)
接下來,我們來探索一下上面的代碼是如何執(zhí)行的。
在這段代碼中,reducer 是 add 。
首先,因?yàn)槲覀兪堑谝淮螆?zhí)行add,所以數(shù)組中的第一個(gè)元素'a'會(huì)被當(dāng)作add的第一個(gè)參數(shù),然后循環(huán)會(huì)從數(shù)組的第二個(gè)元素'b'開始。這一次,'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)并沒有那種和諧的美感。因?yàn)槲覀儼褦?shù)組的第一個(gè)元素,也就是'a'作為初始的param1,然后從數(shù)組的第二個(gè)元素循環(huán)得到param2。
實(shí)際上,我們可以將reduce中的第二個(gè)參數(shù)指定為reducer函數(shù)的param1的初始值,這樣param2就會(huì)從數(shù)組的第一個(gè)元素開始循環(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è)元素開始依次遍歷數(shù)組。
所以我們可以使用這個(gè)語法來重寫我們的第一個(gè)代碼片段。
var arr = ['a', 'b', 'c', 'd', 'e']; function add(x, y) { return x + y; } arr.reduce(add, '')
接下來,我們將進(jìn)入實(shí)際編程章節(jié),體驗(yàn)reduce的強(qiáng)大威力。
1、累加和累積乘法
如果我們想得到數(shù)組中所有元素的總和,你會(huì)怎么做?
一般來說,你可能會(huì)這樣寫:
function accumulation(arr) { let sum = 0; for (let i = 0; i < arr.length; i++) { sum = sum + arr[i]; } return sum; }
當(dāng)然,你可能還有其他的寫法,但是只要使用for循環(huán),代碼就會(huì)顯得多余。
那我們看看上面的累加函數(shù)是做什么的:
- 將初始總和設(shè)置為零
- 取出數(shù)組中的第一個(gè)元素并求和
- 在 sum 中緩存上一步的結(jié)果
- 依次取出數(shù)組中的其他元素,進(jìn)行上述操作
- 返回最終結(jié)果
我們可以看到,當(dāng)我們用文字描述上述步驟時(shí),很明顯它符合reduce的使用。所以我們可以使用reduce來重寫上面的代碼:
function accumulation(arr) { function reducer(x, y) { return x + y } return arr.reduce(reducer, 0); }
如果你習(xí)慣使用箭頭函數(shù),上面的代碼看起來會(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ù)組的最大值和最小值,可以這樣寫:
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 before if (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ì)象來存儲(chǔ)統(tǒng)計(jì)后的頻率,因?yàn)閿?shù)組中的元素可能是對(duì)象類型,而對(duì)象的key只能是字符串或符號(hào)類型。
這里有兩個(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)樽址愋涂梢杂米鲗?duì)象的鍵,所以我們這里不使用 Map。
4、多個(gè)數(shù)組的展平
function Flat(arr = []) { return arr.reduce((t, v) => t.concat(Array.isArray(v) ? Flat(v) : v), []) }
通過reduce依次訪問數(shù)組中的每個(gè)元素。如果我們發(fā)現(xiàn)元素還是一個(gè)數(shù)組,就遞歸調(diào)用 flat 方法。
總結(jié)
以上就是我今天跟你分享的4個(gè)關(guān)于數(shù)組reduce的用法,希望對(duì)你有所幫助。
相關(guān)文章
非html5實(shí)現(xiàn)js版彈球游戲示例代碼
彈球游戲,一般都是使用html5來實(shí)現(xiàn)的,其實(shí)不然,使用js也可以實(shí)現(xiàn)類似的效果,下面有個(gè)不錯(cuò)的示例,感興趣的朋友可以參考下,希望對(duì)大家有所幫助2013-09-09JS圖片自動(dòng)輪換效果實(shí)現(xiàn)思路附截圖
這篇文章主要介紹了JS實(shí)現(xiàn)的圖片自動(dòng)輪換效果,有具體的實(shí)現(xiàn)思路及截圖,希望對(duì)大家學(xué)習(xí)有所幫助2014-04-04動(dòng)態(tài)加載圖片路徑 保持JavaScript控件的相對(duì)獨(dú)立性
根據(jù)新界面的要求,需要一部分圖片來增強(qiáng)日期控件的美觀性??紤]到既要實(shí)現(xiàn)加載圖表的目標(biāo),又要保持控件的獨(dú)立性以便將來的移植。2010-09-09JavaScript組成、引入、輸出、運(yùn)算符基礎(chǔ)知識(shí)講解
JavaScript 被數(shù)百萬計(jì)的網(wǎng)頁用來改進(jìn)設(shè)計(jì)、驗(yàn)證表單、檢測(cè)瀏覽器、創(chuàng)建cookies,以及更多的應(yīng)用。這篇文章主要介紹了JavaScript組成、引入、輸出、運(yùn)算符基礎(chǔ)知識(shí)講解,需要的朋友可以參考下2016-12-12短視頻(douyin)去水印工具的實(shí)現(xiàn)代碼
這篇文章主要介紹了市面上短視頻(douyin)"去水印"的工具原來是這樣實(shí)現(xiàn)的,本文給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2021-03-03ionic中的$ionicPlatform.ready事件中的通用設(shè)置
$ionicPlatform.ready事件是用于檢測(cè)當(dāng)前的平臺(tái)是否就緒的事件,相當(dāng)于基于document的deviceready事件, 在app中一些通用關(guān)于設(shè)備的設(shè)置必須在這個(gè)事件中處理2017-06-06antd-mobile ListView長(zhǎng)列表的數(shù)據(jù)更新遇到的坑
這篇文章主要介紹了antd-mobile ListView長(zhǎng)列表的數(shù)據(jù)更新遇到的坑,文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2020-04-04