return function() {
// Сконструировать список аргументов, начиная со второго аргумента
// метода bind, и передать все эти аргументы указанной функции,
var args = [], і;
fог(і = 1; і < boundArgs.length; i++) args.push(boundArgs[i]);
for(i = 0; і < arguments.length; i++) args.push(arguments[i]);
// Теперь вызвать self как метод объекта о со всеми аргументами
return self.apply(о, args);
};
};
}
Обратите внимание, что функция, возвращаемая этим методом bind(),
является замыканием, использующим переменные self
и boundArgs,
объявленные во внешней функции, которые остаются доступными вложенной функции даже после того, как она будет возвращена внешней функцией и вызвана из-за пределов внешней функции.
Метод bind(),
определяемый стандартом ECMAScript 5, имеет некоторые особенности, которые невозможно реализовать в ECMAScript 3. Прежде всего, настоящий метод bind()
возвращает объект функции, свойство length
которой установлено в соответствии с количеством параметров связываемой функции, минус количество связанных аргументов (но не меньше нуля). Во-вторых, метод bind(
) в ECMAScript 5 может использоваться для частичного применения функций-конструкторов. Если функцию, возвращаемую методом bind(),
использовать как конструктор, значение this
, переданное методу bind(),
игнорируется, и оригинальная функция будет вызвана как конструктор, с уже связанными аргументами, если они были определены. Функции, возвращаемые методом bind(),
не имеют свойства prototype (свойство prototype обычных функций нельзя удалить), и объекты, созданные связанными функциями-конструкторами, наследуют свойство prototype оригинального, несвязанного конструктора. Кроме того, с точки зрения оператора instanceof
связанные конструкторы действуют точно так же, как несвязанные конструкторы.
Подобно другим объектам в языке JavaScript, функции имеют метод toString()
. Спецификация ECMAScript требует, чтобы этот метод возвращал строку, следующую синтаксису инструкции объявления функции. На практике большинство (но не все) реализаций метода toString()
возвращают полный исходный текст функции. Для встроенных функций обычно возвращается строка, содержащая вместо тела функции текст «[native code]» или аналогичный.
8.7.6. Конструктор Function()
Функции обычно определяются с помощью ключевого слова function
либо в форме инструкции объявления функции, либо в форме выражения-литерала. Однако функции могут также определяться с помощью конструктора Function().
Например:
var f = new Function("x", "у", "return x*y;");
Эта строка создаст новую функцию, которая более или менее эквивалентна функции, объявленной с помощью более привычного синтаксиса:
var f = function(x, у) { return х*у; }
Конструктор Function()
принимает произвольное число строковых аргументов. Последний аргумент должен содержать текст с телом функции; он может включать произвольное число инструкций на языке JavaScript, разделенных точкой с запятой. Все остальные аргументы конструктора интерпретируются как имена параметров функции. Чтобы создать функцию, не имеющую аргументов, достаточно передать конструктору всего одну строку - тело функции.
Примечательно, что конструктору Function()
не передается никаких аргументов, определяющих имя создаваемой функции. Подобно литералам функций, конструктор Function()
создает анонимные функции.
Есть несколько моментов, связанных с конструктором Function(),
о которых следует упомянуть особо:
• Конструктор Function()
позволяет динамически создавать и компилировать функции в процессе выполнения программы.
• При каждом вызове конструктор Function()
выполняет синтаксический анализ тела функции и создает новый объект функции. Если вызов конструктора производится в теле цикла или часто вызываемой функции, это может отрицательно сказаться на производительности программы. Напротив, вложенные функции и выражения определения функций внутри циклов не компилируются повторно.
• И последний, очень важный момент: когда функция создается с помощью конструктора Function(),
не учитывается лексическая область видимости - функции всегда компилируются как глобальные функции, что наглядно демонстрирует следующий фрагмент:
Читать дальше
Конец ознакомительного отрывка
Купить книгу