const o = {};
//내부 슬롯은 자바스크립트 엔진의 내부 로직이므로 직접 접근할 수 없다.
o.[[Prototype]] // Unexpected token '['
o.__proto__ // Object.prototype
프로퍼티의 상태란?
프로퍼티의 값(value), 값의 갱신 가능 여부(writable), 열거 가능 여부(enumerable), 재정의 가능 여부(configurable)를 말한다.
const person = {
name: 'Lee'
};
console.log(Object.getOwnPropertyDescriptor(person,'name')
//{value:"Lee", writable: true, enumerable: true, configurable: true}
const person = {
name: "Lee"
};
person.age = 20;
console.log(Object.getOwnPropertyDescriptors(person));
{
name: {value: "Lee", writable:true, ... }
age: { value: 20, writable: true, ... }
}
프로퍼티 어트리뷰트 | 프로퍼티 디스크립터 객체의 프로퍼티 | 설명 |
[[Value]] | value | |
[[Writable]] | writable | |
[[Enumerable]] | enumerable | |
[[Configurable]] | configurable |
프로퍼티 어트리뷰트 | 프로퍼티 디스크립터 객체의 프로퍼티 | 설명 |
[[Get]] | get | |
[[Set]] | set | |
[[Enumerable]] | enumerable | |
[[Configurable]] | configurable |
const person = {
//데이터 프로퍼티
firstName: 'Myungseong',
lastName: 'Kim',
//fullName은 접근자 함수로 구성된 접근자 프로퍼티다.
//getter 함수
get fullName(){
return `${this.firstName} ${this.lastName}`;
},
//setter함수
set fullName(name){
//배열 디스트럭처링 할당 -문자열은 iterable한 유사배열이다.
//띄어쓰기로 나누어진 문자열을 각 프로퍼티 키의 값으로 할당.
[this.firstName, this.lastName] = name.split(' ');
}
};
//데이터 프로퍼티를 통한 프로퍼티 값의 참조
console.log(person.firstName + ' ' + person.lastname); // MyungSeong Kim
//접근자 프로퍼티를 통한 프로퍼티 값의 저장
//접근자 프로퍼티 fullName에 값을 저장하면 setter함수가 호출된다.
person.fullName = 'Heegun Lee';
console.log(person); // {firstName: "Heegun", lastName: "Lee"}
//접근자 프로퍼티를 통한 프로퍼티 값의 참조
// 접근자 프로퍼티 fullName에 접근하면 getter 함수가 호출된다.
console.log(person.fullName); //Heegun Lee
//firstName은 데이터 프로퍼티다.
//데이터 프로퍼티는 4가지의 프로퍼티 어트리뷰트를 갖는다.
let descriptor = Object.getOwnPropertyDescriptor(person,"firstname");
console.log(descriptor);
//output { value: "Heegun", writable: "true", ...}
//fullName은 접근자 프로퍼티다.
//접근자 프로퍼티도 4가지의 프로퍼티 어트리뷰트를 갖는다.
descriptor = Objectg.getOwnPropertyDescriptor(person,'fullName');
console.log(descriptor);
//output { get: f, set:f, ...}
person 객체의 firstName과 lastName 프로퍼티는 일반적인 데이터 프로퍼티다.
get,set메서드는 getter,setter 함수이고, getter/setter 함수의 이름 fullName이 접근자 프로퍼티다.
접근자 프로퍼티는 자체적으로 값(프로퍼티 어트리뷰트[[Value]])을 가지지 않으며 다만 데이터 프로퍼티의 값을 읽거나(get) 저장할 때(set) 관여할 뿐이다.
프로토타입 (prototype)
프로토타입은 어떤 객체의 상위 객체의 역할을 하는 객체다. 프로토타입은 하위 객체에게 자신의 프로퍼티와 메서드를 상속한다.
프로토타입의 프로퍼티,메서드를 상속받은 하위 객체는 자신의 프로퍼티,메서드처럼 자유롭게 사용할 수 있다.프로토타입 체인은 프로토타입이 단방향으로 연결된 상속 구조를 말한다.
객체의 프로퍼티나 메서드에 접근하려고 할 때, 해당 객체에 접근하려는 프로퍼티 또는 메서드가 없다면 프로토타입 체인을 따라 검색한다.
const person = {};
Object.defineProperty(person,'firstName',{
value: 'MyungSeong',
writable: true,
enumerable: true,
configurable:true
});
Object.defineProperty(person,'lastName',{
value: 'Kim'
});
// 디스크립터 객체의 프로퍼티를 누락하고 정의한 lastName은 undefined,false가 기본값이된다.)
let descriptor = Object.getOwnPropertyDescriptor(person,'lastName');
console.log('lastName',descriptor);
// lastName {value: "Kim", writable: false, enumerable: false configurable : false}
//[[Enumerable]]
// Enumerable이 false인 경우 for...in, Object.keys 등으로 열거할 수 없다.
// lastName property는 [[Enumerable]]값이 false이므로 열거되지 않는다.
console.log(Object.keys(person)); // ["firstName"]
//[Writable]]
// Writable이 false인 경우 해당 프로퍼티의 Value 값을 변경할 수 없다.
// lastName은 false이므로 값이 변경되지 않는다. 에러는 발생하지 않고 무시된다.
person.lastName = "Lee"; // 무시된다.
//[[Configurable]]
//Configurable이 false인 경우 값을 삭제하거나 재정의 할 수 없다.
// lastName은 false이므로 값이 변경되지 않는다. 에러는 발생하지 않고 무시된다.
delete person.lastName; // 무시된다.
Object.defineProperty(person,'lastName',{ enumerable: true });
// output : Cannot redefine property: lastName
_
// 접근자 프로퍼티 정의
Object.defineProperty(person, 'fullName', {
get() {
return `${this.firstName} ${this.lastName}`;
},
set(name){
[this.firstName, this.lastName] = name.split(' ');
},
enumerable: true,
configurable: true
});
person.fullName = "Heechan Choi"
console.log(person) // { firstName: "Heechan" lastName: "Choi"}
function deepFreeze(target){
//객체가 아니거나 동결된 객체는 무시하고 객체이고 동결되지 않은 객체만 동결한다
if(target&& typeof target === 'object' && !Object.isFrozen(target)){
Object.freeze(target);
Object.keys(target).forEach(key => deepFreeze(taget[key]));
}
return target;
}
const person = {
name: "Lee",
address: { city: 'Seoul' }
};
//깊은 객체 동결
deepFreeze(person);
console.log(Object.isFrozen(person));//true
console.log(Object.isFrozen(person.address)); // true
person.address.city = 'Busan';
console.log(person); //{name: "Lee", address: {city: "Seoul"}