Работа с массивами в JavaScript
Массив в JavaScript — упорядоченный набор значений, которые называют элементами, причём каждый элемент имеет свой индекс, характеризующий числовую позицию элемента в массиве. В этой статье мы поговорим, как выполнять работу с массивами в JavaScript.
Для начала вспомним, что в JavaScript массивы являются нетипизированными, то есть элементы массива могут иметь любой тип, при этом разные элементы в одном и том же массиве иногда имеют разные типы. Мало того, элементы массива в JavaScript бывают объектами либо даже другими массивами — всё это позволяет создавать действительно сложные структуры данных, например, массивы массивов и массивы объектов.
Идём дальше. В языке JavaScript отчёт индексов массивов начинается с нуля, причём для них применяются 32-битные целые числа. Что касается первого элемента массива, то его индекс равен нулю.
В JavaScript массивы являются динамическими, в результате чего они способны уменьшаться и увеличиваться в размерах в случае необходимости. И вообще, в JavaScript отсутствует необходимость объявлять фиксированные размеры массивов в процессе их создания либо повторно распределять память в случае изменения размеров массивов.
Также можно сказать, что в JavaScript массивы являются специализированной формой объектов, а индексы массивов значат больше, чем просто имена свойств, по совпадению являющихся целыми числами.
Но давайте закончим со скучной теорией и перейдём к практике. В JavaScript массив представляет объект Array. При этом существует ряд методов и свойств, позволяющих управлять нашим массивом.
Инициализация массива в JavaScript
В JavaScript мы можем создать пустой массив, применяя для этого квадратные скобки либо конструктор Array:
var users = new Array(); var people = []; console.log(users); // Array[0] console.log(people); // Array[0]
Кроме того, мы можем сразу же инициализировать наш массив некоторым числом элементов:
var users = new Array("Bill", "Tom", "Alice"); var people = ["Sam", "John", "Kate"]; console.log(users); // ["Bill", "Tom", "Alice"] console.log(people); // ["John", "Sam", "Kate"]
Также можем определить массив, по ходу определяя в него новые элементы:
var users = new Array(); users[1] = "Tom"; users[2] = "Kate"; console.log(users[1]); // "Tom" console.log(users[0]); // undefined
Обратите внимание, что не имеет значения, что массив по умолчанию создаётся с нулевой длиной. Используя индексы, мы сможем подставить тот или иной элемент на конкретный индекс в массиве.
length
Если хотим узнать длину массива в JS, используем свойство length:
var fruit = new Array(); fruit[0] = "груши"; fruit[1] = "яблоки"; fruit[2] = "сливы"; document.write("В массиве fruit " + fruit.length + " элемента: <br/>"); for(var i=0; i < fruit.length; i++) document.write(fruit[i] + "<br/>");
Фактически, длиной нашего массива в JavaScript будет индекс последнего элемента, плюс единица. К примеру:
var users = new Array(); // в массиве 0 элементов users[0] = "Kate"; users[1] = "Tom"; users[4] = "Sam"; for(var i=0; i<users.length;i++) console.log(users[i]);
В этом случае вывод браузера будет следующим:
Kate Tom undefined undefined Sam
Вне зависимости от того, что для индексов №№ 2 и 3 мы элементов не добавляли, длиной массива будет число 5, ведь элементы с индексами №№ 2 и 3 будут просто иметь значение undefined.
Копирование массива в JavaScript
slice()
В JS копирование массива бывает поверхностным либо неглубоким (shallow copy) а также deep copy, то есть глубоким.
В первом случае мы присваиваем переменной значение другой переменной, хранящей массив:
var users = ["Tom", "Bob", "Bill"]; console.log(users); // ["Tom", "Bob", "Bill"] var people = users; // shallow copy people[1] = "John"; // меняем 2-й элемент console.log(users); // ["Tom", "John", "Bill"]
В нашем случае переменная people после неглубокого копирования станет указывать на тот же массив, что и переменная users. Именно поэтому в случае изменения элементов в people, поменяются элементы и в users, ведь по факту это один и тот же массив.
Вышеописанное поведение не всегда желательно. К примеру, нам надо, чтобы после копирования переменные указывали на отдельные массивы. Тогда подойдёт глубокое копирование посредством метода
var users = ["Tom", "Bob", "Bill"]; console.log(users); // ["Tom", "Bob", "Bill"] var people = users.slice(); // deep copy people[1] = "John"; // меняем 2-й элемент console.log(users); // ["Tom", "Bob", "Bill"] console.log(people); // ["Tom", "John", "Bill"]
Теперь после копирования переменные станут указывать на разные массивы, поэтому мы сможем менять их отдельно друг от друга.
Кроме того, функция
var users = ["Tom", "Bob", "Bill", "Alice", "Kate"]; var people = users.slice(1, 4); console.log(people); // ["Bob", "Bill", "Alice"]
В функцию
push()
Функция
var fruit = []; fruit.push("груши"); fruit.push("яблоки"); fruit.push("сливы"); fruit.push("вишни","абрикосы"); document.write("В массиве fruit " + fruit.length + " элемента: <br/>"); document.write(fruit); // груши,яблоки,сливы,вишни,абрикосы
pop()
Такая функция, как
var fruit = ["груши", "яблоки", "сливы"]; var lastFruit = fruit.pop(); // из массива извлекается последний элемент document.write(lastFruit + "<br/>"); document.write("В массиве fruit " + fruit.length + " элемента: <br/>"); for(var i=0; i <fruit.length; i++) document.write(fruit[i] + "<br/>");
Итоговый вывод:
сливы В массиве fruit 2 элемента: груши яблоки
shift()
Теперь рассмотрим функцию
var fruit = ["груши", "яблоки", "сливы"]; var firstFruit = fruit.shift(); document.write(firstFruit + "<br/>"); document.write("В массиве fruit " + fruit.length + " элемента: <br/>"); for(var i=0; i <fruit.length; i++) document.write(fruit[i] + "<br/>");
Вывод следующий:
груши В массиве fruit 2 элемента: яблоки сливы
unshift()
Что касается функции
var fruit = ["груши", "яблоки", "сливы"]; fruit.unshift("апельсины"); document.write(fruit);
Вывод браузера:
апельсины,груши,яблоки,сливы
Удаление элемента по индексу в JavaScript
splice()
Функция
var users = ["Tom", "Bob", "Bill", "Alice", "Kate"]; var deleted = users.splice(3); console.log(deleted); // [ "Alice", "Kate" ] console.log(users); // [ "Tom", "Bob", "Bill" ]
А возвратит удалённые элементы метод slice.
Как видите, удаление идёт с начала массива. Но если передадим отрицательный индекс, удаление начнётся с конца массива. Давайте удалим лишь последний элемент:
var users = ["Tom", "Bob", "Bill", "Alice", "Kate"]; var deleted = users.splice(-1); console.log(deleted); // [ "Kate" ] console.log(users); // [ "Tom", "Bob", "Bill", "Alice" ]
Кстати, можно задать конечный индекс для удаления. Удалим элементы, начиная с первого, заканчивая третьим индексами:
var users = ["Tom", "Bob", "Bill", "Alice", "Kate"]; var deleted = users.splice(1,3); console.log(deleted); // [ "Bob", "Bill", "Alice" ] console.log(users); // [ "Tom", "Kate" ]
Можно вставить новые элементы вместо удаляемых:
var users = ["Tom", "Bob", "Bill", "Alice", "Kate"]; var deleted = users.splice(1,3, "Ann", "Roy"); console.log(deleted); // [ "Bob", "Bill", "Alice" ] console.log(users); // [ "Tom", "Ann", "Roy", "Kate" ]
Обратите внимание, что мы удалили 3 элемента (с первого по третий индексы), а вместо них вставили только два элемента.
concat()
Метод или функция
var fruit = ["груши", "яблоки", "сливы"]; var vegetables = ["огурцы", "помидоры", "картошка"]; var products = fruit.concat(vegetables); for(var i=0; i < products.length; i++) document.write(products[i] + "<br/>");
И совсем необязательно объединять лишь однотипные массивы, можно и разнотипные:
var fruit = ["груши", "яблоки", "сливы"]; var prices = [20, 30, 70]; var products = fruit.concat(prices);
join()
Функцией
var fruit = ["груши", "яблоки", "сливы", "абрикосы", "апельсины"]; var fruitString = fruit.join(", "); document.write(fruitString);
В функцию
sort()
Функция
var fruit = ["груши", "яблоки", "сливы", "абрикосы", "пони"]; fruit.sort(); for(var i=0; i < fruit.length; i++) document.write(fruit[i] + "<br/>");
В результате элементы будут отсортированы по алфавиту:
абрикосы груши пони сливы яблоки
reverse()
С помощью
var fruit = ["груши", "яблоки", "сливы", "абрикосы", "апельсины"]; fruit.reverse(); for(var i=0; i < fruit.length; i++) document.write(fruit[i] + "<br/>");
Итог:
апельсины абрикосы сливы яблоки груши
Поиск индекса элемента в JS
Функции
var fruit = ["яблоки", "груши", "огурцы", "яблоки", "груши"]; var firstIndex = fruit.indexOf("яблоки"); var lastIndex = fruit.lastIndexOf("яблоки"); var otherIndex = fruit.indexOf("черешня"); document.write(firstIndex); // 0 document.write(lastIndex); // 3 document.write(otherIndex); // -1
У firstIndex значение 0, так как первое включение «яблоки» в нашем массиве приходится на индекс 0, последнее — на индекс № 3. Если же элемент в массиве отсутствует, функции
every()
С помощью
var numbers = [ 1, -12, 8, -2, 25, 62 ]; function condition(value, index, array) { var result = false; if (value > 0) { result = true; } return result; }; var passed = numbers.every(condition); document.write(passed); // false
В метод
function condition(value, index, array) { }
Здесь параметр value представляет перебираемый текущий элемент массива, параметр index представляет индекс данного элемента, а параметр array осуществляет передачу ссылки на массив.
В такой функции можно проверить переданное значение элемента на его соответствие определённому условию. В нашем примере мы проверяем каждый элемент массива на условие, больше ли он нуля. Когда больше, возвращается значение true, так как элемент соответствует условию. Когда меньше, возвращается значение false, т. к. элемент не соответствует нашему условию.
В результате, когда осуществляется вызов метода numbers.every(condition) он выполняет перебор всех элементов нашего массива numbers, а потом поочерёдно передает их в функцию condition. Когда эта функция возвращает значение true для всех элементов, метод
some()
Функция/метод
Здесь
var numbers = [ 1, -12, 8, -2, 25, 62 ]; function condition(value, index, array) { var result = false; if (value === 8) { result = true; } return result; }; var passed = numbers.some(condition); // true
filter()
Как
var numbers = [ 1, -12, 8, -2, 25, 62 ]; function condition(value, index, array) { var result = false; if (value > 0) { result = true; } return result; }; var filteredNumbers = numbers.filter(condition); for(var i=0; i < filteredNumbers.length; i++) document.write(filteredNumbers[i] + "<br/>");
Вот результат вывода:
1 8 25 62
forEach() и map()
Функции
var numbers = [ 1, 2, 3, 4, 5, 6]; for(var i = 0; i<numbers.length; i++){ var result = numbers[i] * numbers[i]; document.write("Квадрат нашего числа " + numbers[i] + " равен " + result + "<br/>"); }
Конструкция может быть упрощена посредством
var numbers = [ 1, 2, 3, 4, 5, 6]; function square(value, index, array) { var result = value * value; document.write("Квадрат нашего числа " + value + " равен " + result + "<br/>"); }; numbers.forEach(square);
Здесь
Что касается
Допустим, давайте, применим map к вычислению квадратов чисел нашего массива:
var numbers = [ 1, 2, 3, 4, 5, 6]; function square(value, index, array) { return result = value * value; }; var squareArray = numbers.map(square); document.write(squareArray);
Функция, передаваемая в