JavaScript數(shù)組reduce()方法使用實(shí)例詳解
前言
今天請讓我詳細(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ì)更簡潔:
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); // 892、獲取一個(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的用法,希望有所幫助。
到此這篇關(guān)于JavaScript數(shù)組reduce()方法使用實(shí)例詳解的文章就介紹到這了,更多相關(guān)JavaScript 數(shù)組reduce方法內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
JavaScript高級(jí)程序設(shè)計(jì) 事件學(xué)習(xí)筆記
JavaScript高級(jí)程序設(shè)計(jì) 事件學(xué)習(xí)筆記,需要的朋友可以參考下。2011-09-09
詳解js中構(gòu)造流程圖的核心技術(shù)JsPlumb
這篇文章主要介紹了js中構(gòu)造流程圖的核心技術(shù)JsPlumb,jsPlumb是一個(gè)強(qiáng)大的JavaScript連線庫,它可以將html中的元素用箭頭、曲線、直線等連接起來,適用于開發(fā)Web上的圖表、建模工具等,需要的朋友可以參考下2015-12-12
IE6背景圖片不緩存問題解決方案及圖片使用策略多個(gè)方法小結(jié)
最近發(fā)現(xiàn)的Web項(xiàng)目在IE6下出現(xiàn)背景圖片不緩存的問題,在網(wǎng)上搜索了一番,發(fā)現(xiàn)已經(jīng)是眾所周知的BUG,這里就總結(jié)一下各種解決辦法2012-05-05
JavaScript實(shí)現(xiàn)上下浮動(dòng)的窗口效果代碼
這篇文章主要介紹了JavaScript實(shí)現(xiàn)上下浮動(dòng)的窗口效果代碼,可實(shí)現(xiàn)自定義窗口在垂直方向上彈性移動(dòng)的效果,代碼備有完整的注釋說明供讀者參考學(xué)習(xí),需要的朋友可以參考下2015-10-10
javascript實(shí)現(xiàn)將文件保存到本地方法匯總
本文給大家匯總介紹了3中使用javascript實(shí)現(xiàn)將文件保存到本地的方法,非常的簡單實(shí)用,有需要的小伙伴可以參考下。2015-07-07
基于JavaScript判斷瀏覽器到底是關(guān)閉還是刷新(超準(zhǔn)確)
這篇文章主要介紹了基于JavaScript判斷瀏覽器到底是關(guān)閉還是刷新(超準(zhǔn)確)的相關(guān)資料,需要的朋友可以參考下2016-02-02
JavaScript tab選項(xiàng)卡插件實(shí)例代碼
這篇文章主要介紹了JavaScript tab選項(xiàng)卡插件實(shí)例代碼的相關(guān)資料,需要的朋友可以參考下2016-02-02
表格展示利器 Bootstrap Table實(shí)例代碼
在表格導(dǎo)出數(shù)據(jù)中,發(fā)現(xiàn)設(shè)置了分頁參數(shù),導(dǎo)出的數(shù)據(jù)僅為表格加載的分頁參數(shù)數(shù)據(jù),于是,針對(duì)這樣的情況,通過設(shè)置分頁參數(shù)的值,使表格可以加載更多的數(shù)據(jù),可達(dá)到導(dǎo)出所有數(shù)據(jù)的功能需求。下面通過本文給大家分享表格展示利器 Bootstrap Table2017-09-09

