내부 슬롯과 내부 메서드
내부 슬롯과 내부 메서드는 자바스크립트 엔진의 구현 알고리즘을 설명하기 위해 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
'개발 기초 > 언어' 카테고리의 다른 글
[JavaScript] 생성자 함수에 의한 객체 생성 (1) | 2025.01.22 |
---|---|
[Java] Priority Queue 클래스 사용자 정의 방법 (0) | 2025.01.22 |
[JavaScript] let, const 키워드와 블록 레벨 스코프 (2) | 2025.01.04 |
[JavaScript] 전역변수의 문제점 (0) | 2025.01.02 |
[JavaScript] 스코프 (0) | 2024.12.31 |