개발 기초/언어

[JavaScript] 프로퍼티 어트리뷰트

숩니따 2025. 1. 8. 22:07

내부 슬롯과 내부 메서드

내부 슬롯과 내부 메서드는 자바스크립트 엔진의 구현 알고리즘을 설명하기 위해 ECMAScript 사양에서 사용하는 의사 프로퍼티(presudo property)와 의사 메서드(preseudo method)로 ECMAScript 사양에 등장하는 이중 대괄호 ([[...]])로 감싼 이름들이 내부 슬롯과 내부 메서드이다.

다만, 개발자가 직접 접근할 수 있또록 외부로 공개된 객체의 프로퍼티는 아니다.

프로퍼티 어트리뷰트와 프로퍼티 디스크립터 객체

  • 자바스크립트 엔진은 프로퍼티를 생성할 때 프로퍼티의 상태를 나타내는 프로퍼티 어트리뷰트를 기본값으로 정의한다.
  • 프로퍼티 어트리뷰트는 자바스크립트 엔진이 관리하는 내부 상태 값인 내부 슬롯 [[Value]], [[Writable]], [[Enumerable]], [[Configurable]]로 프로퍼티 어트리뷰트에 직접 접근할 수는 없지만 Object.getOwnPropertyDescriptor 메서드 사용하여 간접적으로 확인할 수 있다.
    • 프로퍼티 디스크립터 객체 반환한다.
    • 존재하지 않는 프로퍼티나 상속받은 프로퍼티에 대한 디스크립터를 요구하면 undefined 반환한다.
    • ES8에 도입된 Object.getOwnPropertyDescriptors 메서드는 모든 프로퍼티의 프로퍼티 어트리뷰트 정보 제공하는 프로퍼티 디스크립터 객체들을 반환한다.

데이터 프로퍼티와 접근자 프로퍼티

데이터 프로퍼티

키와 값으로 구성된 일반적인 프로퍼티

| 프로퍼티 어트리뷰트 | 프로퍼티 디스크립터
객체의 프로퍼티
| 설명 |
| --- | --- | --- |
| [[Value]] | value | - 프로퍼티 키를 통해 프로퍼티 값에 접근하면 반환되는 값

  • 프로퍼티 키를 통해 프로퍼티 값을 변경하면 [[value]]에 값을 재할당, 이때 프로퍼티가 없으면 동적 생성 후 값 저장 |
    | [[Writable]] | writable | - 프로퍼티 값의 변경 가능 여부를 나타내며 불리언 값을 가짐
  • 값이 false인 경우 해당 프로퍼티의 값을 변경할 수 없는 읽기 전용 프로퍼티가 됨 |
    | [[Enumerable]] | enumerable | - 프로퍼티의 열거 가능 여부를 나타내며 불리언 값을 가짐
  • 값이 false인 경우 해당 프로퍼티는 for…in문이나 Object.keys 메서드 등으로 열거 불가능 |
    | [[Configurable]] | configurable | - 프로퍼티의 재정의 가능 여부를 나타내며 불리언 값을 가짐
  • 값이 false인 경우 해당 프로퍼티의 삭제, 프로퍼티 어트리뷰트 값의 변경이 금지됨
  • 단, [[Writable]]이 true인 경우 [[Value]]의 변경과 [[Writable]]을 false로 변경하는 것은 허용 |

접근자 프로퍼티

  • 자체적으로 값을 갖지 않고 다른 데이터 프로퍼티의 값을 읽거나 저장할 때 호출되는 접근자 함수로 구성된 프로퍼티이다.
  • 접근자 함수는 getter/setter 함수라고도 부르고 접근자 프로퍼티는 두 함수 모두 정의 가능하고 하나만 정의 가능하다.

| 프로퍼티 어트리뷰트 | 프로퍼티 디스크립터
객체의 프로퍼티
| 설명 |
| --- | --- | --- |
| [[Get]] | get | - 접근자 프로퍼티를 통해 프로퍼티의 값을 읽을 때 호출되는 접근자 함수

  • 접근자 프로퍼티 키로 프로퍼티 값에 접근하면 getter 함수가 호출되고 그 결과가 프로퍼티 값으로 반환 |
    | [[Set]] | set | - 접근자 프로퍼티를 통해 데이터 프로퍼티의 값을 저장할 때 호출되는 접근자 함수
  • 근접자 프로퍼티 키로 프로퍼티 값을 저장하면 setter 함수가 호출되고 그 결과가 프로퍼티 값으로 저장 |
    | [[Enumerable]] | enumerable | - 프로퍼티의 열거 가능 여부를 나타내며 불리언 값을 가짐
  • 값이 false인 경우 해당 프로퍼티는 for…in문이나 Object.keys 메서드 등으로 열거 불가능 |
    | [[Configurable]] | configurable | - 프로퍼티의 재정의 가능 여부를 나타내며 불리언 값을 가짐
  • 값이 false인 경우 해당 프로퍼티의 삭제, 프로퍼티 어트리뷰트 값의 변경이 금지됨
  • 단, [[Writable]]이 true인 경우 [[Value]]의 변경과 [[Writable]]을 false로 변경하는 것은 허용 |

프로퍼티 정의

  • 새로운 프로퍼티를 추가하면서, 프로퍼티 어트리뷰트를 명시적으로 정의하거나, 기존 프로퍼티의 프로퍼티 어트리뷰트를 재정의하는 것이다.
  • Object.defineProperty 메서드를 사용하면 프로퍼티의 어트리뷰트를 정의할 수 있으며, 인수로는 객체의 참조와 데이터 프로퍼티의 키인 문자열, 플퍼티 디스크립터 객체를 전달한다.
  • Object.defineProperty 메서드로 정의할 때 프로퍼티 디스크립터 객체의 프로퍼티를 일부 생략 가능하다.
  • Object.defineProperts 메서드를 사용하면 여러 개의 프로퍼티를 한 번에 정의할 수 있다.

| 프로퍼티디스크립터
객체의 프로퍼티
| 대응하는 프로퍼티 어트리뷰트 | 생략했을 때의 기본값 |
| --- | --- | --- |
| value | [[Value]] | undefined |
| get | [[Get]] | undefined |
| set | [[Set]] | undefined |
| writable | [[Writable]] | false |
| enumerable | [[Enumerable]] | false |
| configurable | [[Configurable]] | false |

객체 변경 방지

  • 객체는 변경 가능한 값으로 재할당 없이 변경할 수 있으므로 프로퍼티를 추가하거나 삭제할 수 있고 프로퍼티 값을 갱신할 수 있다.
  • Object.defineProperty 또는 Object.defineProperties 메서드를 사용하여 프로퍼티 어트리뷰트를 재정의할 수 있다.
  • 자바스크립트는 객체의 변경을 방지하는 다양한 메서드를 제공하는데 메서드들은 객체 변경을 금지하는 강도가 다르다.
구분 메서드 프로퍼티 추가 프로퍼티 삭제 프로퍼티 값 읽기 프로퍼티 값 쓰기 프로퍼티 어트리뷰트 재정의
객체 확장 금지 Object.preventExtensions X O O O O
객체 밀봉 Object.seal X X O O X
객체 동결 Object.freeze X X O X X

객체 확장 금지

  • Object.preventExtensions 메서드는 프로퍼티 추가가 금지된다.
    → 프로퍼티 동적 추가와 Object.defineProperty 메서드로 추가 가능한데 이 두 가지 추가 방법이 모두 금지
  • 확장이 가능한 객체인지 여부는 Object.isExtensible 메서드로 확인할 수 있다.
  • 프로퍼티 추가는 금지되지만 삭제는 가능하다.

객체 밀봉

  • Object.seal 메서드는 프로퍼티 추가 및 삭제와 프로퍼티 어트리뷰트 재정의 금지를 의미한다.
  • 밀봉된 객체는 읽기와 쓰기만 가능하다.
  • 밀봉된 객체인지 여부는 Object.isSealed 메서드로 확인할 수 있다.
  • 프로퍼티 추가와 삭제는 금지되지만 프로퍼티 값 갱신은 가능하다.

객체 동결

  • Object.freeze 메서드는 프로퍼티 추가 및 삭제와 프로퍼티 어트리뷰트 재정의 금지, 프로퍼티 값 갱신 금지를 의미한다.
  • 동결된 객체는 읽기만 가능하다.
  • 동결된 객체인지 여부는 Object.isFrozen 메서드로 확인할 수 있다.

불변 객체

  • 위의 변경 방지 메서드는 얕은 변경 방지로 직속 프로퍼티만 변경이 망지되고 중첩 객체까지는 영향을 주지 못하여 중첩 객체까지 영향을 줄 수 없다.
  • 중첩 객체까지 동결하여 변경이 불가능한 읽기 전용의 불변 객체를 구현하려면 객체를 값으로 갖는 모든 프로퍼티에 대해 재귀적으로 Object.freeze 메서드를 호출해야 한다.
function deepFreeze(target) {
    if (targe t && typeof target === 'object' && !Object.isFrozen(target)) {
        Object.freeze(target);
        // 모든 프로퍼티를 순회하여 재귀적으로 동결
        Object.keys(target).forEach(key => deepFreeze(target[key]));
    }
    return target;
}

const person = {
    name: 'Lee',
    address: { city : 'Seoul'}
};

deepFreeze(person);

[출처] 모던자바스크립트 Deep Dive