Пример 9.18. Неизменяемый класс со свойствами и методами, доступными только для чтения
// Эта функция может работать и без ключевого слова 'new': она одновременно
// является и конструктором, и фабричной функцией
function Range(from,to) {
// Дескрипторы свойств from и to, доступных только для чтения,
var props = {
from: {value:from, enumerable:true,writable:false,configurable:false},
to: {value:to, enumerable:true, writable:false, configurable:false}
};
if (this instanceof Range) // Если вызвана как конструктор
Object.defineProperties(this, props): // Определить свойства
else // Иначе как фабричная функция
return Object.create(Range.prototype, // Создать и вернуть новый
props): // объект Range со свойствами
}
// Если добавлять свойства в объект Range.prototype тем же способом, можно будет
// определить атрибуты этих свойств. Поскольку мы не указываем атрибуты enumerable,
// writable и configurable, они по умолчанию получают значение false.
Object.defineProperties(Range.prototype, {
includes: {
value: function(x) { return this.from <= x && x <= this.to: }
),
foreach: {
value: function(f) {
for(var x = Math.ceil(this.from); x <= this.to; x++) f(x);
}
},
toString: {
value: function() { return "(” + this, from + "..." + this, to + }
}
});
Для определения неизменяемых и неперечислимых свойств в примере 9.18 используются функции Object.defineProperties()
и Object.create().
Они предоставляют широкие возможности, но необходимость определять для них объекты дескрипторов свойств может сделать программный код более сложным для чтения. Чтобы избежать этого, можно определить вспомогательные функции для изменения атрибутов свойств, которые уже были определены. Две такие вспомогательные функции демонстрируются в примере 9.19.
Пример 9.19/ Вспомогательные функции для работы с дескрипторами свойств
// Делает указанные (или все) свойства объекта о
// недоступным для записи и настройки,
function freezeProps(o) {
var props = (arguments.length == 1) // Если один аргумент,
? Object.getOwnPropertyNames(o) // изменить все свойства,
: Array.prototype.splice.call(arguments, 1);
// иначе только указанные
props.forEach(function(n) { // Делает каждое свойство ненастраиваемым
// и доступным только для чтения
// Пропустить ненастраиваемые свойства
if (!Object.getOwnPropertyDescriptor(o,n).configurable) return:
Object.defineProperty(o, n, { writable: false, configurable: false });
}):
return о; // Чтобы можно было продолжить работу с объектом о
}
// Делает неперечислимыми указанные (или все) свойства объекта о,
// если они доступны для настройки,
function hideProps(o) {
var props = (arguments.length == 1) // Если один аргумент,
? Object.getOwnPropertyNames(o) // изменить все свойства,
: Array.prototype.splice.call(arguments, 1);
// иначе только указанные
props.forEach(function(n) { // Скрыть каждое от цикла for/in
// Пропустить ненастраиваемые свойства
if (!Object.getOwnPropertyDescriptor(o,n).configurable) return:
Object.defineProperty(o, n, { enumerable: false });
}):
return o;
}
Функции Object.defineProperty()
и Object.defineProperties()
могут использоваться и для создания новых свойств, и для изменения атрибутов уже существующих свойств. При создании новых свойств все опущенные атрибуты по умолчанию принимают значение false
. Однако при изменении атрибутов уже существующих свойств опущенные атрибуты не изменяются. Например, в функции hideProps()
выше указывается только атрибут enumerable
, потому что функция должна изменять только его.
С помощью этих двух функций можно писать определения классов с использованием преимуществ ECMAScript 5, без существенного изменения привычного стиля определения классов. В примере 9.20 приводится определение неизменяемого класса Range
, в котором используются наши вспомогательные функции.
Читать дальше
Конец ознакомительного отрывка
Купить книгу