最新文章專(zhuān)題視頻專(zhuān)題問(wèn)答1問(wèn)答10問(wèn)答100問(wèn)答1000問(wèn)答2000關(guān)鍵字專(zhuān)題1關(guān)鍵字專(zhuān)題50關(guān)鍵字專(zhuān)題500關(guān)鍵字專(zhuān)題1500TAG最新視頻文章推薦1 推薦3 推薦5 推薦7 推薦9 推薦11 推薦13 推薦15 推薦17 推薦19 推薦21 推薦23 推薦25 推薦27 推薦29 推薦31 推薦33 推薦35 推薦37視頻文章20視頻文章30視頻文章40視頻文章50視頻文章60 視頻文章70視頻文章80視頻文章90視頻文章100視頻文章120視頻文章140 視頻2關(guān)鍵字專(zhuān)題關(guān)鍵字專(zhuān)題tag2tag3文章專(zhuān)題文章專(zhuān)題2文章索引1文章索引2文章索引3文章索引4文章索引5123456789101112131415文章專(zhuān)題3
問(wèn)答文章1 問(wèn)答文章501 問(wèn)答文章1001 問(wèn)答文章1501 問(wèn)答文章2001 問(wèn)答文章2501 問(wèn)答文章3001 問(wèn)答文章3501 問(wèn)答文章4001 問(wèn)答文章4501 問(wèn)答文章5001 問(wèn)答文章5501 問(wèn)答文章6001 問(wèn)答文章6501 問(wèn)答文章7001 問(wèn)答文章7501 問(wèn)答文章8001 問(wèn)答文章8501 問(wèn)答文章9001 問(wèn)答文章9501
當(dāng)前位置: 首頁(yè) - 科技 - 知識(shí)百科 - 正文

關(guān)于JavaScript 數(shù)組你應(yīng)該知道的事情(推薦)

來(lái)源:懂視網(wǎng) 責(zé)編:小采 時(shí)間:2020-11-27 21:58:53
文檔

關(guān)于JavaScript 數(shù)組你應(yīng)該知道的事情(推薦)

關(guān)于JavaScript 數(shù)組你應(yīng)該知道的事情(推薦):首先做一個(gè)粗體聲明:循環(huán)經(jīng)常是無(wú)用的,并且使得代碼很難閱讀。 當(dāng)談到迭代一個(gè)數(shù)組的時(shí)候,無(wú)論你想去查找元素,排序或者任何其他的事,都有可能存在一個(gè)數(shù)組的方法供你使用。 然而,盡管它們有用,但其中一些仍然不被人了解。我會(huì)努力為你展示一些有用的方
推薦度:
導(dǎo)讀關(guān)于JavaScript 數(shù)組你應(yīng)該知道的事情(推薦):首先做一個(gè)粗體聲明:循環(huán)經(jīng)常是無(wú)用的,并且使得代碼很難閱讀。 當(dāng)談到迭代一個(gè)數(shù)組的時(shí)候,無(wú)論你想去查找元素,排序或者任何其他的事,都有可能存在一個(gè)數(shù)組的方法供你使用。 然而,盡管它們有用,但其中一些仍然不被人了解。我會(huì)努力為你展示一些有用的方

首先做一個(gè)粗體聲明:循環(huán)經(jīng)常是無(wú)用的,并且使得代碼很難閱讀。
當(dāng)談到迭代一個(gè)數(shù)組的時(shí)候,無(wú)論你想去查找元素,排序或者任何其他的事,都有可能存在一個(gè)數(shù)組的方法供你使用。

然而,盡管它們有用,但其中一些仍然不被人了解。我會(huì)努力為你展示一些有用的方法。把這篇文章當(dāng)做對(duì) JavaScript 數(shù)組方法的指引吧。

注意: 在開(kāi)始之前,不得不了解一件事:我比較偏愛(ài)函數(shù)式編程。所以我傾向于使用的方法不會(huì)直接改變?cè)瓉?lái)的數(shù)組。這種方法,我避免了副作用。我不是說(shuō)不應(yīng)該改變數(shù)組,但至少要了解那些方法會(huì)改變,那些會(huì)有副作用。副作用導(dǎo)致不想要的改變,而不想要的改變帶來(lái)bugs!

了解到這里,我們可以開(kāi)始正文了。

必不可少的
當(dāng)跟數(shù)組打交道時(shí),有四件事你應(yīng)該清楚:map,filter,reduce 和 展開(kāi)操作符。它們富有力量。

map

你可以在很多種情況下使用它。基本地,每次你需要修改數(shù)組的元素時(shí),考慮使用 map。
它接受一個(gè)參數(shù):一個(gè)方法,在每一個(gè)數(shù)組元素上調(diào)用。然后返回一個(gè)新的數(shù)組,所以沒(méi)有副作用。

const numbers = [1, 2, 3, 4]

const numbersPlusOne = numbers.map(n => n + 1) // 每個(gè)元素 +1
console.log(numbersPlusOne) // [2, 3, 4, 5]

你也能創(chuàng)建一個(gè)新數(shù)組,用于保留對(duì)象的一個(gè)特殊屬性:

const allActivities = [
 { title: 'My activity', coordinates: [50.123, 3.291] },
 { title: 'Another activity', coordinates: [1.238, 4.292] },
 // etc.
]

const allCoordinates = allActivities.map(activity => activity.coordinates)
console.log(allCoordinates) // [[50.123, 3.291], [1.238, 4.292]]

所以,請(qǐng)記住,當(dāng)你需要去轉(zhuǎn)換數(shù)組時(shí),考慮使用map。

filter

這個(gè)方法的名字在這里十分準(zhǔn)確的:當(dāng)你想去過(guò)濾數(shù)組的時(shí)候使用它。
如同map所做,它接受一個(gè)函數(shù)作為它的唯一參數(shù),在數(shù)組的每個(gè)元素上調(diào)用。這個(gè)方法返回一個(gè)布爾值:

  1. true 如果你需要在數(shù)組中保留元素
  2. false 如果你不想保留它

接著你會(huì)得到一個(gè)帶有你想要保留的元素的新數(shù)組。
舉個(gè)例子,你可以在數(shù)組中只保留奇數(shù):

const numbers = [1, 2, 3, 4, 5, 6]
const oddNumbers = numbers.filter(n => n % 2 !== 0)
console.log(oddNumbers) // [1, 3, 5]

或者你可以在數(shù)組中移除特殊的項(xiàng):

const participants = [
 { id: 'a3f47', username: 'john' },
 { id: 'fek28', username: 'mary' },
 { id: 'n3j44', username: 'sam' },
]

function removeParticipant(participants, id) {
 return participants.filter(participant => participant.id !== id)
}

console.log(removeParticipant(participants, 'a3f47')) // [{ id: 'fek28', username: 'mary' }, { id: 'n3j44', username: 'sam' }];

reduce

個(gè)人認(rèn)為是最難理解的方法。但是如果你一旦掌握它,很多瘋狂的事情你都可以用它做到。
基本地, reduce 使用有值的數(shù)組然后組合成一個(gè)新的值。它接受兩個(gè)參數(shù),一個(gè)回調(diào)方法就是我們的 reducer 和一個(gè)可選的初始化的值(默認(rèn)是數(shù)組的第一個(gè)項(xiàng))。這個(gè) reducer 自己使用四個(gè)參數(shù):

  1. 累計(jì):在你的 reducer 中累積的返回值
  2. 當(dāng)前數(shù)組的值
  3. 當(dāng)前索引
  4. 當(dāng)前調(diào)用 reduce 的數(shù)組

大多數(shù)時(shí)候,你只需要使用前兩個(gè)參數(shù):累計(jì)值和當(dāng)前值。
拋開(kāi)這些理論。來(lái)看看常見(jiàn)的一個(gè) reduce 的例子。

const numbers = [37, 12, 28, 4, 9]
const total = numbers.reduce((total, n) => total + n)
console.log(total) // 90

在第一個(gè)遍歷時(shí),這個(gè)累計(jì)值,也就是 total,使用了初始化為 37 的值。它返回的值是 37 + n 并且 n 等于 12,因此得到 49.在第二次遍歷時(shí),累加值是 49,返回值是 49 + 28 = 77。如此繼續(xù)直到第四次。

reduce 是很強(qiáng)大的,你可以實(shí)際使用它去構(gòu)建很多數(shù)組的方法,比如 map 或者 filter:

const map = (arr, fn) => {
 return arr.reduce((mappedArr, element) => {
 return [...mappedArr, fn(element)]
 }, [])
}

console.log(map([1, 2, 3, 4], n => n + 1)) // [2, 3, 4, 5]

const filter = (arr, fn) => {
 return arr.reduce((filteredArr, element) => {
 return fn(element) ? [...filteredArr] : [...filteredArr, element]
 }, [])
}

console.log(filter([1, 2, 3, 4, 5, 6], n => n % 2 === 0)) // [1, 3, 5]

根本上看,我們給 reduce 一個(gè)初始默認(rèn)值 []:我們的累計(jì)值。對(duì)于 map,我們運(yùn)行一個(gè)方法,它的結(jié)果是累加到最后,多虧了 展開(kāi)操作符(不必?fù)?dān)心,后面討論)。對(duì)于 filter,幾乎是相似的,除了我們?cè)谠厣线\(yùn)行過(guò)濾函數(shù)。如果返回 true,我們返回前一個(gè)數(shù)組,否則在數(shù)組最后添加當(dāng)前元素。

我們來(lái)看一個(gè)更高級(jí)的例子:深度展開(kāi)數(shù)組,也就是說(shuō)把 [1, 2, 3, [4, [[[5, [6, 7]]]], 8]] 樣的數(shù)組轉(zhuǎn)換成 [1, 2, 3, 4, 5, 6, 7, 8] 樣的。

function flatDeep(arr) {
 return arr.reduce((flattenArray, element) => {
 return Array.isArray(element)
 ? [...flattenArray, ...flatDeep(element)]
 : [...flattenArray, element]
 }, [])
}

console.log(flatDeep([1, 2, 3, [4, [[[5, [6, 7]]]], 8]])) // [1, 2, 3, 4, 5, 6, 7, 8]

這個(gè)例子有點(diǎn)像 map,除了我們用到了遞歸。我不想去解釋這個(gè)用法,它超出了這篇文章的范圍。但是,如果你想了解更多的關(guān)于遞歸的知識(shí),請(qǐng)參考這篇優(yōu)質(zhì)的文章。

展開(kāi)操作(ES2015)

我知道這不是一個(gè)方法。但是,在處理數(shù)組時(shí),使用展開(kāi)操作可以幫助你做很多事情。事實(shí)上,你可以在另一個(gè)數(shù)組中使用它展開(kāi)一個(gè)數(shù)組的值。從這一點(diǎn)來(lái)說(shuō),你可以復(fù)制一個(gè)數(shù)組,或者連接多個(gè)數(shù)組。

const numbers = [1, 2, 3]
const numbersCopy = [...numbers]
console.log(numbersCopy) // [1, 2, 3]

const otherNumbers = [4, 5, 6]
const numbersConcatenated = [...numbers, ...otherNumbers]
console.log(numbersConcatenated) // [1, 2, 3, 4, 5, 6]

注意::展開(kāi)操作符對(duì)原數(shù)組做了一次淺拷貝。但什么是 淺拷貝?🤔

額,淺拷貝是盡可能少的復(fù)制原數(shù)組。當(dāng)你有一個(gè)數(shù)組包含數(shù)字,字符串或者布爾值(基本類(lèi)型),它們是沒(méi)問(wèn)題的,這些值被真正復(fù)制。然而,對(duì)于 對(duì)象和數(shù)組 而言,這是不同的。只有 對(duì)原值的引用 會(huì)被復(fù)制!因此,如果你創(chuàng)建一個(gè)包含對(duì)象的數(shù)組的淺拷貝,然后在拷貝的數(shù)組中修改了對(duì)象,它也會(huì)修改原數(shù)組的對(duì)象,因?yàn)樗鼈兪?同一個(gè)引用。

const arr = ['foo', 42, { name: 'Thomas' }]
let copy = [...arr]

copy[0] = 'bar'

console.log(arr) // No mutations: ["foo", 42, { name: "Thomas" }]
console.log(copy) // ["bar", 42, { name: "Thomas" }]

copy[2].name = 'Hello'

console.log(arr) // /!\ MUTATION ["foo", 42, { name: "Hello" }]
console.log(copy) // ["bar", 42, { name: "Hello" }]

所以,如果你想去“真正地”靠譜一個(gè)包含對(duì)象或者數(shù)組的是維護(hù)組,你可以使用 lodash 的方法 cloneDeep。但是不要覺(jué)得必須做這樣的事。這里的目標(biāo)是 意識(shí)到事情是如何運(yùn)作的。

最好了解的

下面你看到的方法,是最好了解一下的,同時(shí)它們能幫助你解決某些問(wèn)題,比如在數(shù)組中搜索一個(gè)元素,取出數(shù)組的部分或者更多。

includes(ES2015)

你曾經(jīng)嘗試用過(guò) indexOf 去查找一個(gè)數(shù)組中是否存在某個(gè)東西嗎?這是一個(gè)糟糕的方式對(duì)吧?幸運(yùn)的是,includes 為我們做到了這些。給 includes 一個(gè)參數(shù),然后會(huì)在數(shù)組里面搜索它,如果一個(gè)元素存在的話(huà)。

const sports = ['football', 'archery', 'judo']
const hasFootball = sports.includes('football')
console.log(hasFootball) // true

concat

concat 方法可以用來(lái)合并兩個(gè)或者更多的數(shù)組。

const numbers = [1, 2, 3]
const otherNumbers = [4, 5, 6]

const numbersConcatenated = numbers.concat(otherNumbers)
console.log(numbersConcatenated) // [1, 2, 3, 4, 5, 6]

// You can merge as many arrays as you want
function concatAll(arr, ...arrays) {
 return arr.concat(...arrays)
}

console.log(concatAll([1, 2, 3], [4, 5, 6], [7, 8, 9], [10, 11, 12])) // [1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12]

forEach

無(wú)論何時(shí)你想為數(shù)組的每個(gè)元素執(zhí)行一些事情時(shí),可以使用 forEach。它使用一個(gè)函數(shù)作為參數(shù),然后給它三個(gè)參數(shù):當(dāng)前值,索引,和當(dāng)前數(shù)組。

const numbers = [1, 2, 3, 4, 5]
numbers.forEach(console.log)
// 1 0 [ 1, 2, 3 ]
// 2 1 [ 1, 2, 3 ]
// 3 2 [ 1, 2, 3 ]

indexOf

這個(gè)用來(lái)在給定的數(shù)組中找出第一個(gè)被發(fā)現(xiàn)的元素的索引。 indexOf 也廣泛用于檢查元素是否在一個(gè)數(shù)組中。不過(guò)老實(shí)說(shuō),我如今已經(jīng)不這樣使用了。

const sports = ['football', 'archery', 'judo']

const judoIndex = sports.indexOf('judo')
console.log(judoIndex) // 2

find

find 方法十分類(lèi)似于 filter 方法。你必須提供一個(gè)函數(shù)用于測(cè)試數(shù)組的元素。然而,find 一旦發(fā)現(xiàn)有一個(gè)元素通過(guò)測(cè)試,就立即停止測(cè)試其他元素。不用于 filter,filter 將會(huì)迭代整個(gè)數(shù)組,無(wú)論情況如何。

const users = [
 { id: 'af35', name: 'john' },
 { id: '6gbe', name: 'mary' },
 { id: '932j', name: 'gary' },
]

const user = users.find(user => user.id === '6gbe')
console.log(user) // { id: '6gbe', name: 'mary' }

所以使用 filter,當(dāng)你想去過(guò)濾整個(gè)數(shù)組時(shí)。使用 find 在當(dāng)你確定在數(shù)組中找某個(gè)唯一元素的時(shí)候。

findIndex

這個(gè)方法完全跟 find 相同除了它返回第一個(gè)發(fā)現(xiàn)元素的索引,而不是直接返回元素。

const users = [
 { id: 'af35', name: 'john' },
 { id: '6gbe', name: 'mary' },
 { id: '932j', name: 'gary' },
]

const user = users.findIndex(user => user.id === '6gbe')
console.log(user) // 1

你或許認(rèn)為 findIndex 跟 indexOf 是相同的。額……不完全是。indexOf 的第一個(gè)元素是基本值(布爾,數(shù)字,字符串,null,undefined或者一個(gè) symbol)而findIndex的第一個(gè)元素是一個(gè)回調(diào)方法。
所以當(dāng)你需要搜索在數(shù)組中的一個(gè)元素的基本值時(shí),使用 indexOf。如果有更復(fù)雜的元素,比如object,使用 findIndex。

slice

當(dāng)你需要取出或者復(fù)制數(shù)組的一部分,可以使用 slice。但是注意,像展開(kāi)操作符一樣, slice 返回部分的淺拷貝!

const numbers = [1, 2, 3, 4, 5]
const copy = numbers.slice()

我在文章的開(kāi)始談到,循環(huán)是沒(méi)有什么用的。來(lái)用一個(gè)例子說(shuō)明你如何擺脫它。

假設(shè)你想去從 API 中去除一定量的聊天記錄里,然后展示它們中的 5 條。有兩種方式實(shí)現(xiàn):一種是循環(huán),另一種是 slice。

// 傳統(tǒng)方式
// 用循環(huán)來(lái)決定消息的數(shù)量
const nbMessages = messages.length < 5 ? messages.length : 5
let messagesToShow = []
for (let i = 0; i < nbMessages; i++) {
 messagesToShow.push(posts[i])
}

// 假設(shè) arr 少于 5 個(gè)元素
// slice 將會(huì)返回原數(shù)組的整個(gè)淺拷貝
const messagesToShow = messages.slice(0, 5)

some

如果你想測(cè)試數(shù)組中 至少有一個(gè)元素 通過(guò)測(cè)試,那么可以使用 some。就像是 map,filter,和 find,some 用回調(diào)函數(shù)作為參數(shù)。它返回 ture,如果至少一個(gè)元素通過(guò)測(cè)試,返回 true 否則返回 false。
當(dāng)你處理權(quán)限問(wèn)題的時(shí)候,可以使用 some:

const users = [
 {
 id: 'fe34',
 permissions: ['read', 'write'],
 },
 {
 id: 'a198',
 permissions: [],
 },
 {
 id: '18aa',
 permissions: ['delete', 'read', 'write'],
 },
]

const hasDeletePermission = users.some(user =>
 user.permissions.includes('delete')
)
console.log(hasDeletePermission) // true

every

類(lèi)似 some,不同的是 ever 測(cè)試了所有的元素是否滿(mǎn)足條件(而不是 至少一個(gè))。

const users = [
 {
 id: 'fe34',
 permissions: ['read', 'write'],
 },
 {
 id: 'a198',
 permissions: [],
 },
 {
 id: '18aa',
 permissions: ['delete', 'read', 'write'],
 },
]

const hasAllReadPermission = users.every(user =>
 user.permissions.includes('read')
)
console.log(hasAllReadPermission) // false

flat(ES2019)

這是一個(gè)即將到來(lái)的招牌方法, 在JavaScript 世界中。大致而言,flat 穿件一個(gè)新數(shù)組,通過(guò)組合所有的子數(shù)組元素。接受一個(gè)參數(shù),數(shù)值類(lèi)型,代表你想展開(kāi)的深度。

const numbers = [1, 2, [3, 4, [5, [6, 7]], [[[[8]]]]]]

const numbersflattenOnce = numbers.flat()
console.log(numbersflattenOnce) // [1, 2, 3, 4, Array[2], Array[1]]

const numbersflattenTwice = numbers.flat(2)
console.log(numbersflattenTwice) // [1, 2, 3, 4, 5, Array[2], Array[1]]

const numbersFlattenInfinity = numbers.flat(Infinity)
console.log(numbersFlattenInfinity) // [1, 2, 3, 4, 5, 6, 7, 8]

flatMap(ES2019)

猜猜這個(gè)方法干什么?我打賭你可以做到顧名思義。

首先在每個(gè)元素上運(yùn)行一個(gè) mapping 方法。接著一次性展示數(shù)據(jù)。十分簡(jiǎn)單!

const sentences = [
 'This is a sentence',
 'This is another sentence',
 "I can't find any original phrases",
]

const allWords = sentences.flatMap(sentence => sentence.split(' '))
console.log(allWords) // ["This", "is", "a", "sentence", "This", "is", "another", "sentence", "I", "can't", "find", "any", "original", "phrases"]

這個(gè)例子中,數(shù)組里有一些句子,然而我們想得到所有的單詞。不使用 map 去把所有的句子分割成單詞然后展開(kāi)數(shù)組,你可以直接使用 flatMap。
與 flatMap 無(wú)關(guān)的,你可以使用 reduce 方法來(lái)計(jì)算單詞的數(shù)量(只是展示另一種 reduce 的用法)

const wordsCount = allWords.reduce((count, word) => {
 count[word] = count[word] ? count[word] + 1 : 1
 return count
}, {})
console.log(wordsCount) // { This: 2, is: 2, a: 1, sentence: 2, another: 1, I: 1, "can't": 1, find: 1, any: 1, original: 1, phrases: 1, }

flatMap 經(jīng)常用于響應(yīng)式編程,這里有個(gè)例子。

join

如果你需要基于數(shù)組元素創(chuàng)建字符串,join 正是你所尋找的。它允許通過(guò)鏈接數(shù)組元素來(lái)創(chuàng)建一個(gè)新的字符串,通過(guò)提供的分割符分割。

舉個(gè)例子,你可以使用 join 一眼展示活動(dòng)的參與者:

const participants = ['john', 'mary', 'gary']
const participantsFormatted = participants.join(', ')
console.log(participantsFormatted) // john, mary, gary

下面的例子更真實(shí),在于你想先過(guò)濾參與者然后得到他們的名字。

const potentialParticipants = [
 { id: 'k38i', name: 'john', age: 17 },
 { id: 'baf3', name: 'mary', age: 13 },
 { id: 'a111', name: 'gary', age: 24 },
 { id: 'fx34', name: 'emma', age: 34 },
]

const participantsFormatted = potentialParticipants
 .filter(user => user.age > 18)
 .map(user => user.name)
 .join(', ')

console.log(participantsFormatted) // gary, emma

from

這是一個(gè)靜態(tài)方法,從類(lèi)數(shù)組中創(chuàng)建新的數(shù)組,或者像例子中的字符串一樣遍歷對(duì)象。當(dāng)處理 dom 時(shí),這個(gè)方法十分有用。

const nodes = document.querySelectorAll('.todo-item') // 這是一個(gè) nodeList 實(shí)例
const todoItems = Array.from(nodes) // 現(xiàn)在你能使用 map filter 等等,就像在數(shù)組中那樣!

你曾經(jīng)見(jiàn)到過(guò)我們使用 Array 代替數(shù)組實(shí)例嗎?這就是問(wèn)什么 from 被稱(chēng)作靜態(tài)方法。

接著可以愉快處理這些節(jié)點(diǎn),比如用 forEach 在每個(gè)節(jié)點(diǎn)上注冊(cè)事件監(jiān)聽(tīng):

todoItems.forEach(item => {
 item.addEventListener('click', function() {
 alert(`You clicked on ${item.innerHTML}`)
 })
})

最好了解突變

下面是其他常見(jiàn)的數(shù)組方法。不同之處在于,它們會(huì)修改原數(shù)組。修改數(shù)組并沒(méi)有什么錯(cuò),最好是你應(yīng)該有意識(shí)去修改它。

對(duì)于這些方法,如果你不想去改變?cè)瓟?shù)組,只能在操作前淺拷貝或者深拷貝。

const arr = [1, 2, 3, 4, 5]
const copy = [...arr] // or arr.slice()

sort

是的,sort 修改了原數(shù)組。事實(shí)上,在這里進(jìn)行了數(shù)組元素排序。默認(rèn)的排序方法把所有的元素轉(zhuǎn)換成字符串,然后按照字母表排序它們。

const names = ['john', 'mary', 'gary', 'anna']
names.sort()
console.log(names) // ['anna', 'gary', 'john', 'mary']

如果你有 Python 背景的話(huà),要小心了。使用 sort 在數(shù)字?jǐn)?shù)組中不會(huì)得到你想要的結(jié)果

const numbers = [23, 12, 17, 187, 3, 90]
numbers.sort()
console.log(numbers) // [12, 17, 187, 23, 3, 90] 🤔。

那么如何對(duì)一個(gè)數(shù)組排序?額,sort 接受一個(gè)函數(shù),一個(gè)比較函數(shù)。這個(gè)函數(shù)接受兩個(gè)參數(shù):第一個(gè)元素(我們稱(chēng)呼為 a)和第二個(gè)元素作比較(b)。這兩個(gè)元素之間的比較需要返回一個(gè)數(shù)字。

  1. 如果為負(fù),a 排序在 b 之前。
  2. 如果為正,b 排序在 a 之前。
  3. 如果是0,沒(méi)有任何改變。

那么你可以使用下面的方式排序數(shù)組:

const numbers = [23, 12, 17, 187, 3, 90]
numbers.sort((a, b) => a - b)
console.log(numbers) // [3, 12, 17, 23, 90, 187]

或者通過(guò)最近時(shí)間排序:

const posts = [
 {
 title: 'Create a Discord bot under 15 minutes',
 date: new Date(2018, 11, 26),
 },
 { title: 'How to get better at writing CSS', date: new Date(2018, 06, 17) },
 { title: 'JavaScript arrays', date: new Date() },
]
posts.sort((a, b) => a.date - b.date) // Substracting two dates returns the difference in millisecond between them
console.log(posts)
// [ { title: 'How to get better at writing CSS',
// date: 2018-07-17T00:00:00.000Z },
// { title: 'Create a Discord bot under 15 minutes',
// date: 2018-12-26T00:00:00.000Z },
// { title: 'Learn Javascript arrays the functional way',
// date: 2019-03-16T10:31:00.208Z } ]

fill

fill 修改或者填充了數(shù)組的所有元素,從開(kāi)始索引到結(jié)束索引,使用一個(gè)靜態(tài)值。fill 最有用的作用是使用靜態(tài)值填充一個(gè)新數(shù)組。

// Normally I would have called a function that generates ids and random names but let's not bother with that here.
function fakeUser() {
 return {
 id: 'fe38',
 name: 'thomas',
 }
}

const posts = Array(3).fill(fakeUser())
console.log(posts) // [{ id: "fe38", name: "thomas" }, { id: "fe38", name: "thomas" }, { id: "fe38", name: "thomas" }]

reverse

這個(gè)方法名在這里顯而易見(jiàn)。然而,像留意 sort 那樣,reverse 會(huì)反轉(zhuǎn)數(shù)組的位置。

const numbers = [1, 2, 3, 4, 5]

numbers.reverse()
console.log(numbers) // [5, 4, 3, 2, 1]

你可以替換的方法

終于,在這個(gè)最后的部分,你將發(fā)現(xiàn)改變?cè)瓟?shù)組的方法,同時(shí)可以很容易替換其中一些。我不是說(shuō)你應(yīng)該拋棄這些方法。只是想要你意識(shí)到一些數(shù)組方法有副作用,并且這里有可選擇的其他方法。

push

處理數(shù)組時(shí)這是使用最多的方法。事實(shí)上,push 允許你在數(shù)組中添加一個(gè)或者多個(gè)元素。它也通常基于一個(gè)舊數(shù)組構(gòu)建一個(gè)新數(shù)組。

const todoItems = [1, 2, 3, 4, 5]

const itemsIncremented = []
for (let i = 0; i < items.length; i++) {
 itemsIncremented.push(items[i] + 1)
}

console.log(itemsIncremented) // [2, 3, 4, 5, 6]

const todos = ['Write an article', 'Proofreading']
todos.push('Publish the article')
console.log(todos) // ['Write an article', 'Proofreading', 'Publish the article']

如果你需要像 itemsIncremented 一樣構(gòu)建一個(gè)數(shù)組,很多方法都是機(jī)會(huì),像我們的朋友 map,filter或者reduce。事實(shí)上我們可以使用 map 同樣做到:

const itemsIncremented = todoItems.map(x => x + 1)

并且如果你需要使用 push,當(dāng)你要添加新元素的時(shí)候,展開(kāi)操作符為你撐腰。

const todos = ['Write an article', 'Proofreading']
console.log([...todos, 'Publish the article']) // ['Write an article', 'Proofreading', 'Publish the article']

splice

splice 常常用于作為移除某個(gè)索引元素的方法。你可以同樣使用 filter 做到。

const months = ['January', 'February', 'March', 'April', ' May']

// With splice
months.splice(2, 1) // remove one element at index 2
console.log(months) // ['January', 'February', 'April', 'May']

// Without splice
const monthsFiltered = months.filter((month, i) => i !== 3)
console.log(monthsFiltered) // ['January', 'February', 'April', 'May']

你可能會(huì)想,如果我需要移除多個(gè)元素呢?額,使用 slice:

const months = ['January', 'February', 'March', 'April', ' May']

// With splice
months.splice(1, 3) // remove thirds element starting at index 1
console.log(months) // ['January', 'May']

// Without splice
const monthsFiltered = [...months.slice(0, 1), ...months.slice(4)]
console.log(monthsFiltered) // ['January', 'May']

shift

shift 移除數(shù)組的第一個(gè)元素然后返回它。從功能上來(lái)說(shuō),你可以使用 spread/rest 實(shí)現(xiàn)。

const numbers = [1, 2, 3, 4, 5]

// With shift
const firstNumber = numbers.shift()
console.log(firstNumber) // 1
console.log(numbers) // [2, 3, 4, 5]

// Without shift
const [firstNumber, ...numbersWithoutOne] = numbers
console.log(firstNumber) // 1
console.log(numbersWithoutOne) // [2, 3, 4, 5]

unshift

Unshift 允許你在數(shù)組開(kāi)始添加一個(gè)或者多個(gè)元素。像是 shift, 你可以使用展開(kāi)操作符做同樣的事:

const numbers = [3, 4, 5]

// With unshift
numbers.unshift(1, 2)
console.log(numbers) // [1, 2, 3, 4, 5]

// Without unshift
const newNumbers = [1, 2, ...numbers]
console.log(newNumbers) // [1, 2, 3, 4, 5]

太長(zhǎng)不看版:

  1. 無(wú)論何時(shí)你在數(shù)組上操作時(shí),不要使用 for-loop 也不要重復(fù)造輪子,你想做的可能已經(jīng)有一個(gè)方法在那里。
  2. 大多數(shù)情況,你應(yīng)該使用 map,filter,reduce和展開(kāi)操作符。它們對(duì)開(kāi)發(fā)者來(lái)說(shuō)是最基礎(chǔ)的工具。
  3. 有許多方法需要了解像 slice,some,flatMap等等。記住它們并且在合適的時(shí)候使用它們。
  4. 副作用導(dǎo)致不想要的改變。要清楚哪些方法會(huì)改變你的原始數(shù)組。
  5. slice 和展開(kāi)操作符是淺拷貝。因此,對(duì)象和子數(shù)組將會(huì)共享同一個(gè)引用,小心使用它們。
  6. “舊”的改變數(shù)組的方法可以被新的替換。取決于你想做什么。

以上所述是小編給大家介紹的JavaScript 數(shù)組詳解整合,希望對(duì)大家有所幫助,如果大家有任何疑問(wèn)請(qǐng)給我留言,小編會(huì)及時(shí)回復(fù)大家的。在此也非常感謝大家對(duì)腳本之家網(wǎng)站的支持!

聲明:本網(wǎng)頁(yè)內(nèi)容旨在傳播知識(shí),若有侵權(quán)等問(wèn)題請(qǐng)及時(shí)與本網(wǎng)聯(lián)系,我們將在第一時(shí)間刪除處理。TEL:177 7030 7066 E-MAIL:11247931@qq.com

文檔

關(guān)于JavaScript 數(shù)組你應(yīng)該知道的事情(推薦)

關(guān)于JavaScript 數(shù)組你應(yīng)該知道的事情(推薦):首先做一個(gè)粗體聲明:循環(huán)經(jīng)常是無(wú)用的,并且使得代碼很難閱讀。 當(dāng)談到迭代一個(gè)數(shù)組的時(shí)候,無(wú)論你想去查找元素,排序或者任何其他的事,都有可能存在一個(gè)數(shù)組的方法供你使用。 然而,盡管它們有用,但其中一些仍然不被人了解。我會(huì)努力為你展示一些有用的方
推薦度:
標(biāo)簽: 了解 的事 javascript
  • 熱門(mén)焦點(diǎn)

最新推薦

猜你喜歡

熱門(mén)推薦

專(zhuān)題
Top