javascript

JS: перебираем массив быстрее всех

  • вторник, 26 сентября 2023 г. в 00:00:11
https://habr.com/ru/articles/763276/

Современный JS предоставляет множество способов перебора массива. Но какой из них является наиболее эффективным по скорости?

Чтобы ответить на этот вопрос, мы проведем тесты, перебирая массивы разной длины и вызывая для каждого элемента метод toString().

Рассмотрим основные способы перебора: for, for(reverse), while, do..while, for..in, for..of, for..each.

Важно! Точность результата console.time() сильно зависит от конфигурации вашей системы.

Для начала протестируем скорость каждого способа на массиве из 1000 элементов.

const iterations = 1000;

const array = new Array(iterations).fill(0);

console.log("Длина массива", + array.length);

//for
console.time("for");
for (let i = 0; i < array.length; i++) {
array[i].toString();
}
console.timeEnd("for");

//forReverse
console.time("for(reverse)");
for (let i = array.length - 1; i >= 0; i--) {
array[i].toString();
}
console.timeEnd("for(reverse)");

//while
console.time("while");
let i = 0;
while (i < array.length) {
array[i].toString();
i++;
}
console.timeEnd("while");

//do...while
console.time("do...while");
let j = 0;
do {
array[j].toString();
j++;
} while (j < array.length);
console.timeEnd("do...while");

//for...in
console.time("for...in");
for (let k in array) {
k.toString();
}
console.timeEnd("for...in");

//for...of
console.time("for...of");
for (let l of array) {
l.toString();
}
console.timeEnd("for...of");

//for...each
console.time("for...each");
array.forEach((el) => {
el.toString();
});
console.timeEnd("for...each");

Результат интересный: цикл for значительно уступает другим способам. Но сохранится ли эта тенденция при большей выборке? Чтобы получить более ясную картину, увеличим количество итераций.

После проведения тестирования мы не можем однозначно назвать самый быстрый способ перебора массива, так как победители меняются в зависимости от размера массива. Однако, стандартный for цикл, for(reverse) цикл, while цикл и do...while цикл будут наилучшим выбором, так как они работают практически одинаково быстро в большинстве случаев.

Рекомендуется использовать префиксные операторы (--i, ++i) вместо постфиксных (i--, i++), так как постфиксные операторы требуют создания временной копии переменной, что занимает дополнительное время на вычисление значения выражения. В свою очередь, префиксный оператор изменяет значение переменной непосредственно в момент выполнения операции, без создания временной копии, что делает его более эффективным и быстрым.

Также для улучшения скорости необходимо кэшировать длину массива заранее.

const arrayLength = array.length;
for (let i = 0; i < arrayLength; i++) {
	array[i].toString();
}

Если вас волнует производительность, то следуйте вышеуказанным рекомендациям, но, как всегда, оптимизируйте производительность только тогда, когда это необходимо, так как читаемость и удобство обслуживания часто более важны.