JavaScript數(shù)組reduce()方法使用實(shí)例詳解
前言
今天請讓我詳細(xì)介紹一下這個(gè)方法,希望對你有幫助。
這是 reduce 的基本用法:
var arr = [1, 2, 3]; function reducer(parmar1, parmar2){ } arr.reduce(reducer)
reduce是數(shù)組原型對象上的一個(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會返回最后一個(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'會被當(dāng)作add的第一個(gè)參數(shù),然后循環(huán)會從數(shù)組的第二個(gè)元素'b'開始。這一次,'b' 是 add 的第二個(gè)參數(shù)。
第一次計(jì)算后,我們得到結(jié)果'ab'。此結(jié)果將被緩存并在下一次添加計(jì)算中用作 param1。同時(shí),數(shù)組中的第三個(gè)參數(shù)'c'將用作add的param2。
同樣,reduce 會繼續(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就會從數(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ù)組中所有元素的總和,你會怎么做?
一般來說,你可能會這樣寫:
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),代碼就會顯得多余。
那我們看看上面的累加函數(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ù),上面的代碼看起來會更簡潔:
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í)候,我們在求和的時(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對象而不是對象來存儲統(tǒng)計(jì)后的頻率,因?yàn)閿?shù)組中的元素可能是對象類型,而對象的key只能是字符串或符號類型。
這里有兩個(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)樽址愋涂梢杂米鲗ο蟮逆I,所以我們這里不使用 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的用法,希望有所幫助。
到此這篇關(guān)于JavaScript數(shù)組reduce()方法使用實(shí)例詳解的文章就介紹到這了,更多相關(guān)JavaScript 數(shù)組reduce方法內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
JavaScript 監(jiān)聽textarea中按鍵事件
不同的瀏覽器中的事件監(jiān)聽機(jī)制是不同的,以onKeyPress事件為例。2009-10-10JavaScript實(shí)現(xiàn)數(shù)據(jù)類型的相互轉(zhuǎn)換
這篇文章主要為大家詳細(xì)介紹了JavaScript實(shí)現(xiàn)數(shù)據(jù)類型的相互轉(zhuǎn)換,感興趣的朋友可以參考一下2016-03-03openlayers 3實(shí)現(xiàn)車輛軌跡回放
這篇文章主要為大家詳細(xì)介紹了openlayers 3實(shí)現(xiàn)車輛軌跡回放,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2020-09-09js 禁止選擇功能實(shí)現(xiàn)代碼(兼容IE/Firefox)
有時(shí)候出于某種需要,不希望用戶可以選擇某個(gè)區(qū)域,進(jìn)行下面的操作,這里給出簡單的代碼。2010-04-04JS倒計(jì)時(shí)實(shí)例_天時(shí)分秒
下面小編就為大家?guī)硪黄狫S倒計(jì)時(shí)實(shí)例_天時(shí)分秒。小編覺得挺不錯(cuò)的,現(xiàn)在就分享給大家,也給大家做個(gè)參考。一起跟隨小編過來看看吧2017-08-08JavaScript基礎(chǔ)之AJAX簡單的小demo
這篇文章主要介紹了JavaScript基礎(chǔ)之AJAX簡單的小demo ,代碼簡單易懂,非常不錯(cuò),具有參考借鑒價(jià)值,需要的朋友參考下2017-01-01Express框架詳解app函數(shù)使用實(shí)例
這篇文章主要為大家介紹了Express框架app函數(shù)使用實(shí)例詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2023-03-03JavaScript實(shí)時(shí)更新當(dāng)前的時(shí)間的示例代碼
這篇文章主要介紹了JavaScript實(shí)時(shí)更新當(dāng)前的時(shí)間的示例代碼,本文通過實(shí)例代碼給大家介紹的非常詳細(xì),對大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2020-07-07