Выбрать главу

var sum = function(x,у) { return х + у }; // Возвращает сумму 2 аргументов

// Создать новую функцию, подобную sum, но со связанным значением null

// ключевого слова this и со связанным значением первого аргумента, равным 1.

// Новая функция принимает всего один аргумент,

var succ = sum.bind(null, 1);

succ(2) // => 3: аргумент x связан со значением 1, а 2 передается в арг. у

function f(y.z) { return this.x + у + z }; // Еще одна функция сложения

var g = f.bind({x:1}, 2); // Связать this и у

g(3)     // => 6: this.x - связан с 1, у - связан с 2, а 3 передается в z

В ECMAScript 3 также возможно связывать значение this и выполнять частичное применение. Стандартный метод bind() можно имитировать программным кодом, который приводится в примере 8.5. Обратите внимание, что этот метод сохраняется как Function.prototype.bind, благодаря чему все функции наследуют его. Данный прием подробно рассматривается в разделе 9.4.

Пример 8.5. Метод Function.bind() для ECMAScript 3

if (!Function.prototype.bind) {

Function.prototype.bind = function(o /*, аргументы */) {

    // Сохранить this и arguments в переменных, чтобы их можно было

    // использовать во вложенной функции ниже,

    var self = this, boundArgs = arguments;

    // Возвращаемое значение метода bind() - функция

    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 связанные конструкторы действуют точно так же, как несвязанные конструкторы.

8.7.5. Метод toString()

Подобно другим объектам в языке JavaScript, функции имеют метод toString(). Спецификация ECMAScript требует, чтобы этот метод возвращал строку, следующую синтаксису инструкции объявления функции. На практике большинство (но не все) реализаций метода toString() возвращают полный исходный текст функции. Для встроенных функций обычно возвращается строка, содержащая вместо тела функции текст «[native code]» или аналогичный.

8.7.6. Конструктор Function()

Функции обычно определяются с помощью ключевого слова function либо в форме инструкции объявления функции, либо в форме выражения-литерала. Однако функции могут также определяться с помощью конструктора Function(). Например:

var f = new Function("x", "у", "return x*y;");