Ниже приводится пример генератора массивов, использующего созданную выше функцию range() для инициализации массива, содержащего квадраты четных чисел, меньшие 100:
let evensquares = [х*х for (х in range(0,10)) if (х % 2 === 0)]
Эта строка примерно эквивалентна следующим пяти строкам:
let evensquares = [];
for(x in range(0,10)) {
if (x % 2 === 0)
evensquares.push(x*x);
}
В общем случае синтаксис генераторов массивов имеет следующий вид:
[ выражение for ( переменная in объект ) if ( условное выражение ) ]
Обратите внимание на три основные части в квадратных скобках:
• Цикл for/in
или for/each
без тела. Эта часть генератора массивов включает переменную (или несколько переменных при использовании присваивания с разложением) слева от ключевого слова in
и объект (который может быть генератором, итерируемым объектом или массивом) справа от ключевого слова in
. Несмотря на отсутствие тела цикла, эта часть генератора массивов выполняет итерации и присваивает последовательные значения, определяемые переменной. Обратите внимание, что перед именем переменной не допускается указывать ключевое слово var
или let
- генераторы массивов неявно используют ключевое слово let
, а используемая переменная недоступна за пределами квадратных скобок и не затирает существующую переменную с тем же именем.
• После итерируемого объекта может присутствовать ключевое слово if
и условное выражение. Если оно присутствует, условное выражение используется для фильтрации значений, по которым выполняются итерации. Условное выражение вычисляется после получения каждого значения, воспроизводимого циклом for
. Если результатом выражения является false
, это значение пропускается и в массив ничего не добавляется. Ключевое слово if
можно не указывать - если оно отсутствует, генератор массива действует так, как если бы в нем присутствовала конструкция if (true)
.
• Выражение, стоящее перед ключевым словом for
, представляет собой эквивалент тела цикла. После того как значение, возвращаемое итератором, будет присвоено переменной и пройдет проверку условным выражением, будет вычислено значение этого выражения, и полученный результат будет добавлен в создаваемый массив.
Ниже приводятся несколько более конкретных примеров, которые помогут лучше понять синтаксис:
data = [2,3,4, -5]; // Массив чисел
squares = [х*х for each (х in data)]; // Квадраты всех чисел: [4,9,16,25]
// Извлечь квадратные корни из всех неотрицательных элементов
roots = [Math.sqrt(x) for each (x in data) if (x >= 0)]
// Создать массив с именами свойств объекта
о = {а:1, b:2, f: function(){}}
let allkeys = [p for (p in o)]
let ownkeys = [p for (p in o) if (o.hasOwnProperty(p))]
let notfuncs = [k for ([k,v] in Iterator(o)) if (typeof v !== "function")]
11.4.5. Выражения-генераторы
В JavaScript 1.8 [23] На момент написания этих строк выражения-генераторы не поддерживались в Rhino.
можно заменить квадратные скобки в генераторах массивов круглыми скобками и получить выражения-генераторы. Выражение-генератор похоже на генератор массивов (синтаксис в круглых скобках в точности соответствует синтаксису в квадратных скобках), но его значением является объект генератора, а не массив. Преимущество выражений-генераторов перед генераторами массивов в том, что они используют прием отложенных вычислений - вычисления выполняются по мере необходимости, а не все сразу - и позволяют обрабатывать даже бесконечные последовательности. Недостаток генераторов состоит в том, что они обеспечивают только последовательный доступ к своим элементам. То есть, в отличие от массивов, генераторы не позволяют обращаться к элементам по индексам: чтобы получить n-е значение, придется выполнить n-1 итераций.
Ранее в этой главе мы реализовали функцию map():
// Функция-генератор, возвращающая f(х) для каждого элемента х итерируемого объекта і
function map(і. f) {
fоr(let x in і) yield f(x);
}
Выражения-генераторы позволяют избежать необходимости создавать или использовать такую функцию map(). Чтобы получить новый генератор h, возвращающий f (х) для каждого значения х, возвращаемого генератором g, достаточно использовать такой программный код:
Читать дальше
Конец ознакомительного отрывка
Купить книгу