9.5.2. Свойство constructor
Другой способ определения класса объекта заключается в использовании свойства constructor. Поскольку конструкторы считаются именами классов, определение класса выполняется очень просто. Например:
function typeAndValue(x) {
if (х == null) return // Значения null и undefined не имеют конструкт.
switch(x.constructor) {
case Number: return "Number: " + x; // Работает с простыми типами
case String: return "String: " + x + ;
case Date: return "Date: ” + x; // Со встроенными типами
case RegExp: return "Regexp: " + x;
case Complex: return "Complex: + x; // И с пользовательскими типами
}
}
Обратите внимание, что выражения в этом примере, следующие за ключевыми словами case, являются функциями. Если бы мы использовали оператор typeof
или извлекали значение атрибута class
объекта, они были бы строками.
Для приема, основанного на использовании свойства constructor
, характерны те же проблемы, что и для приема на основе оператора instanceof
. Он не всегда будет работать при наличии нескольких контекстов выполнения (например, при наличии нескольких фреймов в окне броузера), совместно использующих общие значения. Каждый фрейм имеет собственный набор функций-конструкторов: конструктор Array в одном фрейме не будет считаться идентичным конструктору Array в другом фрейме.
Кроме того, язык JavaScript не требует, чтобы каждый объект имел свойство constructor
: это всего лишь соглашение, по которому по умолчанию объект-прототип создается для каждой функции, и очень просто по ошибке или преднамерение опустить свойство constructor
в прототипе. Например, первые два класса в этой главе (в примерах 9.1 и 9.2) были определены так, что их экземпляры не имеют свойства constructor
.
Основная проблема использования оператора instanceof
или свойства constructor
для определения класса объекта проявляется при наличии нескольких контекстов выполнения и, соответственно, при наличии нескольких копий функций-конструкторов. Эти функции могут быть совершенно идентичными, но разными объектами, как следствие, не равными друг другу.
Одно из возможных решений проблемы заключается в том, чтобы использовать в качестве идентификатора класса имя функции-конструктора вместо самой функции. Конструктор Array
в одном окне не будет равен конструктору Array
в другом окне, но их имена будут равны. Некоторые реализации JavaScript обеспечивают доступ к имени функции через нестандартное свойство name
объекта функции. Для реализаций, где свойство name
отсутствует, можно преобразовать функцию в строку и извлечь имя из нее. (Этот прием использовался в разделе 9.4, где демонстрировалось добавление в класс Function
метода getName()
.)
В примере 9.4 определяется функция type(),
возвращающая тип объекта в виде строки. Простые значения и функции она обрабатывает с помощью оператора typeof
. Для объектов она возвращает либо значение атрибута class
, либо имя конструктора. В своей работе функция type()
использует функцию classof()
из примера 6.4 и метод Function.getName()
из раздела 9.4. Для простоты реализация этой функции и метода включена в пример.
Пример 9.4. Функция type() для определения типа значения
/**
* Возвращает тип значения в виде строки:
* -Если о - null, возвращает "null", если о - NaN, возвращает "пап”.
* -Если typeof возвращает значение, отличное от "object", возвращает это значение.
* (Обратите внимание, что некоторые реализации идентифицируют объекты
* регулярных выражений как функции.)
* -Если значение атрибута class объекта о отличается от "Object",
* возвращает это значение.
* -Если о имеет свойство constructor, а конструктор имеет имя, возвращает
* имя конструктора.
* -Иначе просто возвращает "Object".
**/
function type(o) {
var t, c, n; // type, class, name
// Специальный случай для значения null:
if (о === null) return "null":
Читать дальше
Конец ознакомительного отрывка
Купить книгу