Методы массива JS
Массивы предоставляют множество методов. Чтобы было проще, в этой главе они разбиты на группы.
Добавление/удаление элементов
Метод splice( )
splice
Как удалить элемент из массива?
Так как массивы – это объекты, то можно попробовать delete:
Вроде бы, элемент и был удалён, но при проверке оказывается, что массив всё ещё имеет 3 элемента arr.length == 3
.
Это нормально, потому что всё, что делает delete obj.key
– это удаляет значение с данным ключом key
. Это нормально для объектов, но для массивов мы обычно хотим, чтобы оставшиеся элементы сдвинулись и заняли освободившееся место. Мы ждём, что массив станет короче.
Поэтому для этого нужно использовать специальные методы.
Метод arr.splice(str)
– это универсальный «швейцарский нож» для работы с массивами. Умеет всё: добавлять, удалять и заменять элементы.
Его синтаксис:
Он начинает с позиции index
, удаляет deleteCount
элементов и вставляет elem1, ..., elemN
на их место. Возвращает массив из удалённых элементов.
Этот метод проще всего понять, рассмотрев примеры.
Начнём с удаления:
Легко, правда? Начиная с позиции 1, он убрал 1 элемент.
В следующем примере мы удалим 3 элемента и заменим их двумя другими.
Здесь видно, что splice
возвращает массив из удалённых элементов:
Метод splice
также может вставлять элементы без удаления, для этого достаточно установить deleteCount
в 0
:
Отрицательные индексы разрешены
В этом и в других методах массива допускается использование отрицательного индекса. Он позволяет начать отсчёт элементов с конца, как тут:
Метод slice( )
Метод arr.slice
намного проще, чем похожий на него arr.splice.
Его синтаксис:
Он возвращает новый массив, в который копирует элементы, начиная с индекса start
и до end
(не включая end). Оба индекса start
и end
могут быть отрицательными. В таком случае отсчёт будет осуществляться с конца массива.
Это похоже на строковый метод str.slice
, но вместо подстрок возвращает подмассивы.
Например:
Можно вызвать slice
и вообще без аргументов: arr.slice()
создаёт копию массива arr. Это часто используют, чтобы создать копию массива для дальнейших преобразований, которые не должны менять исходный массив.
Метод concat( )
Метод arr.concat
создаёт новый массив, в который копирует данные из других массивов и дополнительные значения.
Его синтаксис:
Он принимает любое количество аргументов, которые могут быть как массивами, так и простыми значениями.
В результате мы получаем новый массив, включающий в себя элементы из arr
, а также arg1, arg2
и так далее…
Если аргумент argN
– массив, то все его элементы копируются. Иначе скопируется сам аргумент.
Например:
Обычно он просто копирует элементы из массивов. Другие объекты, даже если они выглядят как массивы, добавляются как есть:
…Но если объект имеет специальное свойство Symbol.isConcatSpreadable
, то он обрабатывается concat
как массив: вместо него добавляются его числовые свойства.
Для корректной обработки в объекте должны быть числовые свойства и length
:
Метод forEach( )
Метод arr.forEach
позволяет запускать функцию для каждого элемента массива.
Его синтаксис:
Например, этот код выведет на экран каждый элемент массива:
А этот вдобавок расскажет и о своей позиции в массиве:
Результат функции (если она вообще что-то возвращает) отбрасывается и игнорируется.
Поиск в массиве
Методы indexOf/lastIndexOf и includes
Далее рассмотрим методы, которые помогут найти что-нибудь в массиве. Методы arr.indexOf, arr.lastIndexOf и arr.includes имеют одинаковый синтаксис и делают по сути то же самое, что и их строковые аналоги, но работают с элементами вместо символов:
ищет arr.indexOf(item, from)
item
, начиная с индексаfrom
, и возвращает индекс, на котором был найден искомый элемент, в противном случае-1
.то же самое, но ищет справа налево. arr.lastIndexOf(item, from)
ищет arr.includes(item, from)
item
, начиная с индексаfrom
, и возвращаетtrue
, если поиск успешен. Например:
Обратите внимание, что методы используют строгое сравнение ===
. Таким образом, если мы ищем false
, он находит именно false
, а не ноль
.
Если мы хотим проверить наличие элемента, и нет необходимости знать его точный индекс, тогда предпочтительным является arr.includes
.
Кроме того, очень незначительным отличием includes
является то, что он правильно обрабатывает NaN
в отличие от indexOf/lastIndexOf
:
Методы find и findIndex
Представьте, что у нас есть массив объектов. Как нам найти объект с определённым условием?
Здесь пригодится метод arr.find
.
Его синтаксис таков:
Функция вызывается по очереди для каждого элемента массива:
item
– очередной элемент.index
– его индекс.array
– сам массив.
Если функция возвращает true
, поиск прерывается и возвращается item
. Если ничего не найдено, возвращается undefined
.
Например, у нас есть массив пользователей, каждый из которых имеет поля id
и name
. Попробуем найти того, кто с id == 1
:
В реальной жизни массивы объектов – обычное дело, поэтому метод find
крайне полезен.
Обратите внимание, что в данном примере мы передаём find
функцию item => item.id == 1
, с одним аргументом. Это типично, дополнительные аргументы этой функции используются редко.
Метод arr.findIndex
– по сути, то же самое, но возвращает индекс, на котором был найден элемент, а не сам элемент, и -1, если ничего не найдено.
Метод filter( )
Метод find
ищет один (первый попавшийся) элемент, на котором функция-колбэк вернёт true
.
На тот случай, если найденных элементов может быть много, предусмотрен метод arr.filter(fn)
.
Синтаксис этого метода схож с find
, но filter
возвращает массив из всех подходящих элементов:
Например:
Преобразование массива
Перейдём к методам преобразования и упорядочения массива.
Метод map( )
Метод arr.map
является одним из наиболее полезных и часто используемых.
Он вызывает функцию для каждого элемента массива и возвращает массив результатов выполнения этой функции.
Синтаксис:
Например, здесь мы преобразуем каждый элемент в его длину:
Метод sort( )
Вызов arr.sort()
сортирует массив на месте, меняя в нём порядок элементов.
Он возвращает отсортированный массив, но обычно возвращаемое значение игнорируется, так как изменяется сам arr.
Например:
Не заметили ничего странного в этом примере?
Порядок стал 1, 15, 2
. Это неправильно! Но почему?
По умолчанию элементы сортируются как строки.
Буквально, элементы преобразуются в строки при сравнении. Для строк применяется лексикографический порядок, и действительно выходит, что "2" > "15"
.
Чтобы использовать наш собственный порядок сортировки, нам нужно предоставить функцию в качестве аргумента arr.sort()
.
Функция должна для пары значений возвращать:
Например, для сортировки чисел:
Теперь всё работает как надо.
Давайте возьмём паузу и подумаем, что же происходит. Упомянутый ранее массив arr
может быть массивом чего угодно, верно? Он может содержать числа, строки, объекты или что-то ещё. У нас есть набор каких-то элементов. Чтобы отсортировать его, нам нужна функция, определяющая порядок, которая знает, как сравнивать его элементы. По умолчанию элементы сортируются как строки.
Метод arr.sort(fn)
реализует общий алгоритм сортировки. Нам не нужно заботиться о том, как он работает внутри (в большинстве случаев это оптимизированная быстрая сортировка). Она проходится по массиву, сравнивает его элементы с помощью предоставленной функции и переупорядочивает их. Всё, что остаётся нам, это предоставить fn
, которая делает это сравнение.
Кстати, если мы когда-нибудь захотим узнать, какие элементы сравниваются – ничто не мешает нам вывести их на экран:
В процессе работы алгоритм может сравнивать элемент с другими по нескольку раз, но он старается сделать как можно меньше сравнений.
Функция сравнения может вернуть любое число
На самом деле от функции сравнения требуется любое положительное число, чтобы сказать «больше», и отрицательное число, чтобы сказать «меньше».
Это позволяет писать более короткие функции:
Лучше использовать стрелочные функции
Помните стрелочные функции?
Можно использовать их здесь для того, чтобы сортировка выглядела более аккуратной:
Будет работать точно так же, как и более длинная версия выше.
Метод reverse( )
Метод arr.reverse
меняет порядок элементов в arr
на обратный.
Например:
Он также возвращает массив arr
с изменённым порядком элементов.
Методы split и join
Ситуация из реальной жизни. Мы пишем приложение для обмена сообщениями, и посетитель вводит имена тех, кому его отправить, через запятую: Вася, Петя, Маша. Но нам-то гораздо удобнее работать с массивом имён, чем с одной строкой. Как его получить?
Метод str.split(delim)
именно это и делает. Он разбивает строку на массив по заданному разделителю delim
.
В примере ниже таким разделителем является строка из запятой и пробела.
У метода split
есть необязательный второй числовой аргумент – ограничение на количество элементов в массиве. Если их больше, чем указано, то остаток массива будет отброшен. На практике это редко используется:
Разбивка по буквам
Вызов split(s)
с пустым аргументом s
разбил бы строку на массив букв:
Вызов arr.join(glue)
делает в точности противоположное split
. Он создаёт строку из элементов arr
, вставляя glue
между ними.
Например:
Метод reduce( )
Если нам нужно перебрать массив – мы можем использовать forEach
, for
или for..of
.
Если нам нужно перебрать массив и вернуть данные для каждого элемента – мы используем map
.
Методы arr.reduce
и arr.reduceRight
похожи на методы выше, но они немного сложнее. Они используются для вычисления какого-нибудь единого значения на основе всего массива.
Синтаксис:
Функция применяется по очереди ко всем элементам массива и «переносит» свой результат на следующий вызов.
Аргументы:
previousValue
– результат предыдущего вызова этой функции, равен initial при первом вызове (если передан initial),item
– очередной элемент массива,index
– его индекс,array
– сам массив.
При вызове функции результат её вызова на предыдущем элементе массива передаётся как первый аргумент.
Звучит сложновато, но всё становится проще, если думать о первом аргументе как «аккумулирующем» результат предыдущих вызовов функции. По окончании он становится результатом reduce
.
Этот метод проще всего понять на примере.
Тут мы получим сумму всех элементов массива всего одной строкой:
Здесь мы использовали наиболее распространённый вариант reduce
, который использует только 2 аргумента.
Давайте детальнее разберём, как он работает.
При первом запуске sum равен initial
(последний аргумент reduce), то есть 0, а current – первый элемент массива
, равный 1. Таким образом, результат функции равен 1.
При втором запуске sum = 1
, и к нему мы добавляем второй элемент массива (2)
.
При третьем запуске sum = 3
, к которому мы добавляем следующий элемент, и так далее…
Здесь отчётливо видно, как результат предыдущего вызова передаётся в первый аргумент следующего.
Мы также можем опустить начальное значение:
Результат – точно такой же! Это потому, что при отсутствии initial
в качестве первого значения берётся первый элемент массива, а перебор стартует со второго.
Таблица вычислений будет такая же за вычетом первой строки.
Но такое использование требует крайней осторожности. Если массив пуст, то вызов reduce
без начального значения выдаст ошибку.
Вот пример:
Поэтому рекомендуется всегда указывать начальное значение.
Метод arr.reduceRight
работает аналогично, но проходит по массиву справа налево.
Метод Array.isArray()
Массивы не образуют отдельный тип языка. Они основаны на объектах.
Поэтому typeof
не может отличить простой объект от массива:
Но массивы используются настолько часто, что для этого придумали специальный метод: Array.isArray(value)
. Он возвращает true, если value массив
, и false, если нет
.
Большинство методов поддерживают «thisArg»
Почти все методы массива, которые вызывают функции – такие как find
, filter
, map
, за исключением метода sort
, принимают необязательный параметр thisArg
.
Этот параметр не объяснялся выше, так как очень редко используется, но для наиболее полного понимания темы мы обязаны его рассмотреть.
Вот полный синтаксис этих методов:
Значение параметра thisArg
становится this
для func
.
Например, вот тут мы используем метод объекта army
как фильтр, и thisArg
передаёт ему контекст:
Если бы мы в примере выше использовали просто users.filter(army.canJoin)
, то вызов army.canJoin
был бы в режиме отдельной функции, с this=undefined
. Это тут же привело бы к ошибке.
Вызов users.filter(army.canJoin, army)
можно заменить на users.filter(user => army.canJoin(user))
, который делает то же самое. Последняя запись используется даже чаще, так как функция-стрелка более наглядна.