JavaScript. Подробное руководство, 6-е издание, стр. 71

С методом

<b>Object.сreate(),</b>
определяемым стандартом ECMAScript 5, мы познакомились в разделе 6.1, где узнали, что первым аргументом этому методу передается объект, который будет служить прототипом для вновь созданного объекта. Этот метод также принимает второй необязательный аргумент, такой же, как и второй аргумент метода
<b>Object.defineProperties().</b>
Если методу
<b>Object.create()</b>
передать множество дескрипторов свойств, они будут использованы для создания свойств нового объекта.

Методы

<b>Object.defineProperty()</b>
и
<b>Object.defineProperties()</b>
возбуждают исключение
<b>ТуреError</b>
, когда создание или изменение свойств запрещено. Например, при попытке добавить новое свойство в нерасширяемый объект (раздел 6.8.3). Другие причины, по которым эти методы могут возбудить исключение
<b>ТуреЕrror</b>
, имеют непосредственное отношение к атрибутам. Атрибут
<b>writable</b>
контролирует попытки изменить атрибут
<b>value</b>
. А атрибут
<b>configurable</b>
контролирует попытки изменить другие атрибуты (а также определяет возможность удаления свойства). Однако все не так просто. Например, значение свойства, доступного только для чтения, можно изменить, если это свойство доступно для настройки. Кроме того, свойство, доступное только для чтения, можно сделать доступным для записи, даже если это свойство недоступно для настройки. Ниже приводится полный перечень правил. Вызовы
<b>Object.defineProperty()</b>
или
<b>Object.defineProperties(),</b>
нарушающие их, возбуждают исключение
<b>ТуреЕrror</b>
:

• Если объект нерасширяемый, можно изменить существующие собственные свойства этого объекта, но нельзя добавить в него новые свойства.

• Если свойство недоступно для настройки, нельзя изменить его атрибуты configurable и enumerable.

• Если свойство с методами доступа недоступно для настройки, нельзя изменить его методы чтения и записи и нельзя превратить его в простое свойство с данными.

• Если свойство с данными недоступно для настройки, нельзя превратить его в свойство с методами доступа.

• Если свойство с данными недоступно для настройки, нельзя изменить значение его атрибута

<b>writable</b>
с
<b>false</b>
на
<b>true</b>
, но его можно Изменить с
<b>true</b>
на
<b>false</b>
.

• Если свойство с данными недоступно для настройки и для записи, нельзя изменить его значение. Однако изменить значение свойства, недоступного для записи можно, если оно доступно для настройки (потому что свойство можно сделать доступным для записи, изменить его значение и затем опять сделать свойство доступным только для чтения).

Пример 6.2 включает функцию

<b>extend(),</b>
которая копирует свойства из одного объекта в другой. Эта функция просто копирует имена и значения свойств и игнорирует их атрибуты. Кроме того, она не копирует методы чтения и записи из свойств с методами доступа, а просто преобразует их в свойства со статическими данными. В примере 6.3 показана новая версия
<b>extend(),</b>
которая копирует все атрибуты свойств с помощью
<b>Object.getOwnPropertyDescriptor()</b>
и
<b>Object.defineProperty().</b>
Но на этот раз данная версия оформлена не как функция, а как новый метод объекта и добавляется в
<b>Object.prototype</b>
как свойство, недоступное для перечисления.

Пример 6.3. Копирование атрибутов свойств

<b>/*</b>

<b>* Добавляет неперечислимый метод extend() в Object.prototype.</b>

<b>* Этот метод расширяет объекты возможностью копирования свойств из объекта,</b>

<b>* переданного в аргументе. Этот метод копирует не только значение свойств,</b>

<b>* но и все их атрибуты. Из объекта в аргументе копируются все собственные</b>

<b>* свойства (даже недоступные для перечисления), за исключением одноименных</b>

<b>* свойств, имеющихся в текущем объекте.</b>

<b>*/</b>

<b>Object.defineProperty(Object.prototype,</b>

<b>  &quot;extend&quot;, // Определяется Object.prototype.extend</b>

<b>  {</b>

<b>    writable: true,</b>

<b>    enumerable: false, // Сделать неперечислимым </b>

<b>    configurable: true,</b>

<b>    value: function(o) { // Значением свойства является данная функция</b>

<b>    // Получить все собственные свойства, даже неперечислимые</b>

<b>    var names = Object.getOwnPropertyNames(o);</b>

<b>    // Обойти их в цикле</b>

<b>    for(var і = 0: і &lt; names.length; i++) {</b>

<b>      // Пропустить свойства, уже имеющиеся в данном объекте </b>

<b>      if (names[i] in this) continue;</b>

<b>      // Получить дескриптор свойства из о</b>

<b>      var desc = Object.getOwnPropertyDescriptor(o,names[i]);</b>

<b>      // Создать с его помощью свойство в данном объекте</b>

<b>      Object.defineProperty(this, names[i], desc);</b>

<b>    }</b>

<b>  }</b>

<b>});</b>

6.7.1. Устаревшие приемы работы с методами чтения и записи

Синтаксис определения свойств с методами доступа в литералах объектов, описанный разделе 6.6, позволяет определять свойства с методами в новых объектах, но, он не дает возможности получать методы чтения и записи и добавлять новые свойства с методами доступа к существующим объектам. В ECMAScript 5 для этих целей можно использовать

<b>Object.getOwnPropertyDescriptor()</b>
и
<b>Object.defineProperty()</b>
.