, представляет собой эквивалент тела цикла. После того как значение, возвращаемое итератором, будет присвоено переменной и пройдет проверку условным выражением, будет вычислено значение этого выражения, и полученный результат будет добавлен в создаваемый массив.
Ниже приводятся несколько более конкретных примеров, которые помогут лучше понять синтаксис:
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")]
можно заменить квадратные скобки в генераторах массивов круглыми скобками и получить выражения-генераторы. Выражение-генератор похоже на генератор массивов (синтаксис в круглых скобках в точности соответствует синтаксису в квадратных скобках), но его значением является объект генератора, а не массив. Преимущество выражений-генераторов перед генераторами массивов в том, что они используют прием отложенных вычислений - вычисления выполняются по мере необходимости, а не все сразу - и позволяют обрабатывать даже бесконечные последовательности. Недостаток генераторов состоит в том, что они обеспечивают только последовательный доступ к своим элементам. То есть, в отличие от массивов, генераторы не позволяют обращаться к элементам по индексам: чтобы получить n-е значение, придется выполнить n-1 итераций.
23
На момент написания этих строк выражения-генераторы не поддерживались в Rhino.
Ранее в этой главе мы реализовали функцию map:
// Функция-генератор, возвращающая f(х) для каждого элемента х итерируемого объекта і
function map(і. f) {
fоr(let x in і) yield f(x);
}
Выражения-генераторы позволяют избежать необходимости создавать или использовать такую функцию map. Чтобы получить новый генератор h, возвращающий f (х) для каждого значения х, возвращаемого генератором g, достаточно использовать такой программный код:
let h = (f(x) for (x in g));
Используя генератор eachline из примера 11.1, можно реализовать отсечение пробельных символов, а также фильтрацию комментариев и пустых строк, как показано ниже:
let lines = eachline(text);
let trimmed = (l.trim for (1 in lines));
let nonblank = (1 for (1 in trimmed) if (1.length > 0 && 1[0]!='#'));
появилась возможность краткой записи простых функций (называется «лексическим замыканием»). Если функция вычисляет единственное выражение и возвращает его значение, ключевое слово
return
и фигурные скобки, окружающие тело функции, можно опустить и просто поместить выражение сразу после списка аргументов. Например:
24
На момент написания этих строк данная особенность не поддерживалась в Rhino.
let succ = function(x) х+1, yes = function true, no = function false;
Это просто и удобно: функции, определяемые таким способом, ведут себя как обычные функции, в определении которых присутствуют фигурные скобки и ключевое слово
return
. Этот сокращенный синтаксис удобно использовать, в частности, при передаче функций другим функциям. Например:
// Отсортировать массив в обратном порядке
data.sort(function(a,b) b-a);
// Определение функции, которая возвращает сумму квадратов элементов массива