Предотвратить такую замену можно, объявив методы экземпляров доступными только для чтения. Сделать это можно с помощью вспомогательной функции freezeProps(),
объявленной выше. Другой способ добиться этого эффекта заключается в использовании функции Object.freeze(),
которая выполняет те же действия, что и функция Object.seal(),
и дополнительно делает все свойства ненастраиваемыми и доступными только для чтения.
Свойства, доступные только для чтения, обладают одной особенностью, о которой необходимо помнить при работе с классами. Если объект о
наследует свойство р
, доступное только для чтения, попытка присвоить значение свойству о.р
будет завершаться неудачей без создания нового свойства в объекте о
. Если потребуется переопределить унаследованное свойство, доступное только для чтения, можно воспользоваться функциями Object.defineProperty(), Object.defineProperties()
или Object.create(),
чтобы создать новое свойство. Это означает, что, когда методы экземпляров класса делаются доступными только для чтения, это существенно осложняет возможность их переопределения в подклассах.
На практике обычно не требуется блокировать возможность изменения объектов-прототипов таким способом, но в некоторых случаях предотвращение расширения объектов может оказаться полезным. Вспомните фабричную функцию enumeration()
из примера 9.7. Она сохраняет все экземпляры перечислений в свойствах объекта-прототипа и в свойстве-массиве values
конструктора. Эти свойства и массив играют роль официального перечня экземпляров перечислений, и их определенно имеет смысл зафиксировать, чтобы исключить возможность добавления новых экземпляров и изменения или удаления существующих. Для этого достаточно добавить в функцию enumeration()
следующие строки:
Object.freeze(enumeration.values);
Object.freeze(enumeration);
Обратите внимание, что применение функции Object.freeze()
к типу перечисления исключает возможность использования свойства objectId
, как было показано в примере 9.17. Решение этой проблемы состоит в том, чтобы прочитать значение свойства objectId
(вызвать соответствующий метод чтения и установить внутреннее свойство) перечисления только один раз, перед тем как его зафиксировать.
9.8.5. Подклассы и ECMAScript 5
В примере 9.22 демонстрируется порядок создания подклассов с использованием возможностей ECMAScript 5. В нем определяется класс stringSet,
наследующий класс AbstractWritableSet
из примера 9.16. Основная особенность этого примера заключается в использовании функции Object.сreate()
для создания объекта-прототипа, наследующего прототип суперкласса, и в определении свойств вновь созданного объекта. Как уже отмечалось выше, основная сложность этого подхода заключается в необходимости использовать неудобные дескрипторы свойств.
Другой интересной особенностью этого примера является передача значения null функции Object.сreate()
при создании объекта, не наследующего ничего. Этот объект используется для хранения элементов множества, а тот факт, что он не имеет прототипа, позволяет вместо метода hasOwnProperty()
использовать оператор in
.
Пример 9.22. StringSet
: определение подкласса множества с использованием ECMAScript 5
function StringSet() {
this.set = Object.create(null); // Создать объект без прототипа
this.n = 0;
this.add.apply(this, arguments);
}
// Обратите внимание, что Object.create позволяет обеспечить наследование
// прототипа суперкласса и определить методы за счет единственного вызова.
// Поскольку при создании свойств мы не указываем значения атрибутов writable,
// enumerable и configurable, они по умолчанию получают значение false.
// Доступность методов только для чтения усложняет их переопределение в подклассах.
StringSet.prototype = Object.create(AbstractWritableSet.prototype, {
constructor: { value: StringSet },
contains: { value: function(x) { return x in this.set; } }.
size: { value: function(x) { return this.n; } },
foreach: { value: function(f,c) { Object.keys(this.set).forEach(f.c); } }.
add: {
value: function() {
Читать дальше
Конец ознакомительного отрывка
Купить книгу