JavaScript中forEach和map詳細講解
基本用法
forEach
和map
都是JavaScript中常見的用于遍歷數(shù)組的方法,它們都可以用于對數(shù)組中的每個元素執(zhí)行某種操作,但它們的使用方式和返回結(jié)果略有不同。
forEach
forEach
是一個用于數(shù)組的方法,它會對數(shù)組中的每個元素執(zhí)行一次回調(diào)函數(shù)。該方法沒有返回值,它只是遍歷整個數(shù)組并執(zhí)行回調(diào)函數(shù)。
forEach
方法的語法如下:
array.forEach(callback(currentValue, index, array), thisArg);
其中,callback
是要執(zhí)行的函數(shù),它會被傳入三個參數(shù):
currentValue
:當前處理的元素。index
:當前處理元素的索引。array
:當前正在遍歷的數(shù)組。
thisArg
是一個可選參數(shù),用于指定執(zhí)行回調(diào)函數(shù)時使用的 this
值。
forEach
方法返回的是undefined
,它不會生成新的數(shù)組。
forEach
方法和for
循環(huán)在功能上類似,但它更簡潔、易讀,也更安全,因為它會避免循環(huán)時的一些常見問題,如數(shù)組越界、重復計算等。
下面是一個使用 forEach
方法遍歷數(shù)組并輸出每個元素的例子:
const arr = [1, 2, 3]; arr.forEach((element) => { console.log(element); });
map
map
方法也是一個用于數(shù)組的方法,它會對數(shù)組中的每個元素執(zhí)行一次回調(diào)函數(shù),并將回調(diào)函數(shù)的返回值存儲在一個新的數(shù)組中。
map
方法的語法如下:
array.map(callback(currentValue, index, array), thisArg);
其中,callback
是要執(zhí)行的函數(shù),它會被傳入三個參數(shù):
currentValue
:當前處理的元素。index
:當前處理元素的索引。array
:當前正在遍歷的數(shù)組。
thisArg
是一個可選參數(shù),用于指定執(zhí)行回調(diào)函數(shù)時使用的 this 值。
map
方法會返回一個新的數(shù)組,這個數(shù)組的長度和原數(shù)組相同,但是每個元素都是經(jīng)過回調(diào)函數(shù)處理后的結(jié)果。
下面是一個使用 map
方法將數(shù)組中的每個元素都加上 1 的例子:
const arr = [1, 2, 3]; const newArr = arr.map((element) => { return element + 1; }); console.log(newArr); // 輸出[2, 3, 4]
可以看到,map
方法生成了一個新的數(shù)組,并將原數(shù)組中的每個元素都加上了 1。
剖析原理
forEach
和map
方法的原理都是基于數(shù)組的迭代器,實際上它們都是通過調(diào)用迭代器實現(xiàn)對數(shù)組的遍歷。
JavaScript數(shù)組有一個內(nèi)置的迭代器,即Array.prototype[@@iterator]
方法。該方法返回一個迭代器對象,可以使用該對象對數(shù)組進行迭代。forEach
和map
方法都會調(diào)用Array.prototype[@@iterator]
方法返回迭代器對象,然后使用該迭代器對象進行遍歷。
forEach
當調(diào)用forEach
方法時,JavaScript引擎會執(zhí)行以下步驟:
- 檢查數(shù)組是否存在,如果不存在,則拋出
TypeError
異常。 - 檢查第一個參數(shù)是否是一個函數(shù),如果不是,則拋出
TypeError
異常。 - 檢查第二個參數(shù)是否是一個對象,如果是,則將該對象作為this值傳遞給回調(diào)函數(shù),否則
this
值為undefined
。 - 獲取迭代器對象,通過調(diào)用
Array.prototype[@@iterator]
方法獲得。 - 對于迭代器對象中的每個元素,依次執(zhí)行回調(diào)函數(shù),并將當前元素的值、索引和原數(shù)組作為參數(shù)傳遞給回調(diào)函數(shù)。
forEach
方法不會返回任何值。
下面是一個簡化的forEach
實現(xiàn)代碼:
Array.prototype.myForEach = function(callback, thisArg) { if (typeof callback !== 'function') { throw new TypeError(callback + ' is not a function'); } for (var i = 0; i < this.length; i++) { callback.call(thisArg, this[i], i, this); } };
可以看到,該實現(xiàn)代碼使用了for
循環(huán)來遍歷數(shù)組,并使用call
方法調(diào)用回調(diào)函數(shù),并將當前元素的值、索引和原數(shù)組作為參數(shù)傳遞給回調(diào)函數(shù)。
map
map
方法的原理與forEach
方法類似,也是通過調(diào)用迭代器實現(xiàn)對數(shù)組的遍歷,只不過map
方法在遍歷時會將回調(diào)函數(shù)的返回值存儲在一個新的數(shù)組中。下面我們以map
方法為例來剖析其原理。
當調(diào)用map
方法時,JavaScript引擎會執(zhí)行以下步驟:
- 檢查數(shù)組是否存在,如果不存在,則拋出
TypeError
異常。 - 檢查第一個參數(shù)是否是一個函數(shù),如果不是,則拋出
TypeError
異常。 - 檢查第二個參數(shù)是否是一個對象,如果是,則將該對象作為
this
值傳遞給回調(diào)函數(shù),否則this
值為undefined
。 - 創(chuàng)建一個新的空數(shù)組,用于存儲回調(diào)函數(shù)的返回值。
- 獲取迭代器對象,通過調(diào)用
Array.prototype[@@iterator]
方法獲得。 - 對于迭代器對象中的每個元素,依次執(zhí)行回調(diào)函數(shù),并將當前元素的值、索引和原數(shù)組作為參數(shù)傳遞給回調(diào)函數(shù)。
- 將回調(diào)函數(shù)的返回值存儲在新數(shù)組中。
- 返回新數(shù)組。
下面是一個簡化的map
實現(xiàn)代碼:
Array.prototype.myMap = function(callback, thisArg) { if (typeof callback !== 'function') { throw new TypeError(callback + ' is not a function'); } var newArray = []; for (var i = 0; i < this.length; i++) { newArray.push(callback.call(thisArg, this[i], i, this)); } return newArray; };
可以看到,該實現(xiàn)代碼除了使用for
循環(huán)遍歷數(shù)組之外,還創(chuàng)建了一個新的數(shù)組,并將回調(diào)函數(shù)的返回值存儲在新數(shù)組中,最后返回新數(shù)組。
forEach和map的共同點
- 都只能遍歷數(shù)組
- 都是循環(huán)遍歷數(shù)組中的每一項。
- 每次執(zhí)行匿名函數(shù)都支持三個參數(shù),參數(shù)分別為item(當前每一項),index(索引值),arr(原數(shù)組)。
- 匿名函數(shù)中的this都是指向window。
forEach和map的不同點
forEach()方法沒有返回值,會更改原數(shù)組。
map() 有返回值,返回一個新數(shù)組,不會改變原數(shù)組,map() 不會對空數(shù)組進行檢測。
總的來說 map 的速度大于 forEach()
性能上來說for > forEach > map
總結(jié)
總的來說,forEach
和map
方法都是基于數(shù)組的迭代器實現(xiàn)的,它們的主要區(qū)別在于對回調(diào)函數(shù)的處理方式。forEach
方法執(zhí)行回調(diào)函數(shù)時,不會將回調(diào)函數(shù)的返回值存儲起來,而是直接忽略。map
方法執(zhí)行回調(diào)函數(shù)時,會將回調(diào)函數(shù)的返回值存儲在一個新數(shù)組中,最后將該數(shù)組返回。因此,如果我們需要對數(shù)組中的每個元素執(zhí)行一個操作,并將操作的結(jié)果存儲在新數(shù)組中,那么應該使用map
方法。如果我們只是需要對數(shù)組中的每個元素執(zhí)行一些操作,而不需要將操作的結(jié)果存儲起來,那么應該使用forEach
方法。
到此這篇關(guān)于JavaScript中forEach和map詳細講解的文章就介紹到這了,更多相關(guān)js forEach和map內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
JavaScript實現(xiàn)圖片懶加載的三種方案詳解
圖片懶加載,當圖片出現(xiàn)在可視區(qū)域再進行加載,提升用戶的體驗,這篇文章主要為大家整理了三個常用的圖片懶加載實現(xiàn)方法,希望對大家有所幫助2023-12-12js實現(xiàn)帶關(guān)閉按鈕始終顯示在網(wǎng)頁最底部工具條的方法
這篇文章主要介紹了js實現(xiàn)帶關(guān)閉按鈕始終顯示在網(wǎng)頁最底部工具條的方法,是非常實用的javascript固定效果,具有一定參考借鑒價值,需要的朋友可以參考下2015-03-03微信小程序?qū)崿F(xiàn)圖片上傳、刪除和預覽功能的方法
這篇文章主要介紹了微信小程序?qū)崿F(xiàn)圖片上傳、刪除和預覽功能的方法,涉及微信小程序界面布局、事件響應及圖片操作相關(guān)實現(xiàn)技巧,需要的朋友可以參考下2017-12-12利用XMLHTTP傳遞參數(shù)在另一頁面執(zhí)行并刷新本頁
利用XMLHTTP傳遞參數(shù)在另一頁面執(zhí)行并刷新本頁...2006-10-10在Monaco Editor中實現(xiàn)斷點設置的方法詳解
Monaco Editor 是 vscode 等產(chǎn)品使用的代碼編輯器,功能強大(且復雜),由微軟維護,本文在 React + TypeScript(Vite)框架下使用 @monaco-editor/react 并介紹開發(fā)斷點顯示時踩到的坑,文中有詳細的代碼示例供大家參考,需要的朋友可以參考下2024-04-04