1 오브젝트의 프로퍼티의 값을 불변하게 해보자
2 오브젝트의 숨은 프로퍼티에 대해 알아보자
3 프로토타입
4 OOP를 프로토타입으로 하는것, 클래스로 하는것 다른가?
5 this를 알아보자

JS OOP를 프로토타입으로 하는것, 클래스로 하는것 다른가?

let list = new Array();
// 이렇게 Array 생성자함수를 이용해서 배열객체를 만들 수 있다

class Erray {

}
// 이렇게 생성자함수라는걸 내가 직접 만들수도 있다

console.log(typeof Erray)
// 함수라고 한다

let erray = new Erray();
console.log(erray.constructor);
// 이렇게 내가 만든 생성자함수로 객체를 만들어서 constructor 를 확인해본다
// 그러면 Erray 를 가리킨다

console.log(erray.constructor === Erray);
// 이렇게 같음을 확인해볼 수 있다

console.log(Object.getPrototypeOf(erray).constructor === Erray);
// 마찬가지로 이렇게도 확인이 가능하다

console.log(Object.getPrototypeOf(erray))
// 프로토타입의 모습도 확인해보자
// 확인결과 constructor 하나 달랑 있다

Erray.prototype.newone = 123;
// 이렇게 값을 넣어준후

console.log(Object.getPrototypeOf(erray))
// 다시 확인해보자
// newone이 들어가있음이 확인된다

console.log(erray.newone)
// 그래서 이렇게 접근이 가능하다

console.log(erray.hasOwnProperty('newone'));
// 하지만 newone은 erray의 속성인것은 아니라 false가 나온다
// 그런데 잠깐.. erray.hasOwnProperty 는 왜 되는걸까? 생각해보자

console.log(Object.getPrototypeOf(erray))
// 분명 여기에는 newone, constructor만 있을뿐이고 erray의 자체속성에도 hasOwnProperty가 없다

console.log(erray['hasOwnProperty'])
// 해보면 나온다

// 자 그 이유는 다음과 같다
console.log(Object.getPrototypeOf(erray))
// 일단 erray의 프로토타입을 확인하고자 한다면 아래와 같이 하면 프로토타입을 볼 수 있다

console.log(Object.getPrototypeOf(Object.getPrototypeOf(erray)))
// 그런데 이것의 프로토타입을 또 보자
// 여기에 hasOwnProperty 가 있는것이 확인된다

console.log(Object.getPrototypeOf(Object.getPrototypeOf(Object.getPrototypeOf(erray))))
// 또 씌우면? 이제는 null이 나온다

class Brray {
   name = 'hello'
   static type = 'Hello'
}
let brray = new Brray();
console.log(brray.name)
console.log(brray.hasOwnProperty('name'))
// 이런 모습의 생성자를 만들어서 객체를 만들어보자
// brray.name 값이 잘 확인된다
// brray.hasOwnProperty 로 name 을 확인해봐도 true로써 속성이 존재함이 확인된다
// static키워드를 붙인 속성인 type은 brray객체와는 전혀 상관없는 요소가 된다.

console.log(Brray.type);
console.log(Brray.hasOwnProperty('type'));
console.log(Object.keys(Brray));
console.log(Object.getOwnPropertyDescriptor(Brray, 'type'))
// 이것에 접근하는 방법은 Brray.type이다
// Brray.hasOwnProperty 로 확인했을때 true로써 type은 Brray의 객체에 포함된 속성으로 확인되고
// Object.keys 로도 확인된다. Object.keys로 확인된다는 것은 enumerable이 true라는 뜻이고 Object.getOwnPropertyDescriptor로 정말 그런지 확인해본결과
// 맞는것으로 확인된다

// 즉 static키워드를 붙인것은
Brray.type2 = 'World'
// 이런식으로 추가된것과 같다고 볼 수 있다


class Crray extends Array {
}
let crray = new Crray();
console.log(crray['forEach']);
// Array를 상속받는 Crray를 만들어보자
// 그리고 forEach 속성을 확인해본 결과 접근이 가능하다
// 일단 forEach는 crray의 직접적 속성으로 존재하는것은 아니다

console.log(Object.getPrototypeOf(crray) === Crray.prototype)
console.log(Object.getPrototypeOf(Object.getPrototypeOf(crray)) === Array.prototype)
console.log(Object.getPrototypeOf(Object.getPrototypeOf(Object.getPrototypeOf(crray))) === Object.prototype)
// 위와 같은 방법으로 확인해보면 모두 true가 나오는것을 확인할 수 있다
// extends 한다는것은 prototype이 연결되게 됨을 확인할 수 있다
// Array또한 extends Object를 하고있는 모습으로 확인된다
// 그렇기 때문에 Object.prototype 에 있는 모든 함수는 crray 를 통해서도 [] 를 통해서도 접근이 가능한것이다

console.log(crray.hasOwnProperty);
console.log([].hasOwnProperty);
console.log(crray.hasOwnProperty === Object.prototype.hasOwnProperty);
// 이렇게말이다

class Drray {
   name = 'dd'
   method() {
      return 'hello';
   }
}
let drray = new Drray();
console.log(drray.method())
// 위와 같이 method 메소드를 만들어보자
// 그리고 이것은 drray.method() 이런식으로 접근이 가능하다

console.log(drray.hasOwnProperty('method'));
console.log(Object.getOwnPropertyNames(drray))
// 이 method란 것이 drray 객체의 속성으로 들어가있을지 확인해봤다. hasOwnProperty로는 false가 나온다
// Object.getOwnPropertyNames 로는 name만 확인된다.
// method는 어디에 있는걸까
// 프로토타입을 확인해보자

console.log(Object.getPrototypeOf(drray))
// 여기에 들어가있음이 확인된다

// 즉
Drray.prototype.method2 = function method2() { return 'hello2'; }
console.log(drray.method2())
// 해서 만든것과 비슷하다는 의미가 된다.
// 이렇게 method2 를 추가한 후 prototype을 확인해보자

console.log(Object.getPrototypeOf(drray))
// 자 이렇게 확인해보자
// method, method2, constructor가 확인되는데 여기서 method2만 enumerable이 true로 보인다.
// 정말 그런지 확인해보자

console.log(Object.getOwnPropertyDescriptor(Drray.prototype, 'method'))
console.log(Object.getOwnPropertyDescriptor(Drray.prototype, 'method2'))
console.log(Object.getOwnPropertyDescriptor(Drray.prototype, 'constructor'))
// 확인 결과 method2의 enumerable은 true임이 확인된다

Object.defineProperty(Drray.prototype, 'method3', {
   value: function method3() { return 'hello3'; },
   enumerable: false,
   configurable: true,
   writable: true,
});
console.log(Object.getOwnPropertyDescriptor(Drray.prototype, 'method'))
console.log(Object.getOwnPropertyDescriptor(Drray.prototype, 'method2'))
console.log(Object.getOwnPropertyDescriptor(Drray.prototype, 'method3'))
console.log(Object.getOwnPropertyDescriptor(Drray.prototype, 'constructor'))
// 위와 같이 추가를 하면 모습이 같아지는 것을 확인 할 수 있다
// 참고로 enumerable, configurable, writable 값을 지정을 안해주면 기본은 false이다
// 그러므로 위 상황에서 enumerable는 false로 할것이라면 안써줘도 상관없었다
let list = new Array(); // 이렇게 Array 생성자함수를 이용해서 배열객체를 만들 수 있다 class Erray { } // 이렇게 생성자함수라는걸 내가 직접 만들수도 있다 console.log(typeof Erray) // 함수라고 한다 let erray = new Erray(); console.log(erray.constructor); // 이렇게 내가 만든 생성자함수로 객체를 만들어서 constructor 를 확인해본다 // 그러면 Erray 를 가리킨다 console.log(erray.constructor === Erray); // 이렇게 같음을 확인해볼 수 있다 console.log(Object.getPrototypeOf(erray).constructor === Erray); // 마찬가지로 이렇게도 확인이 가능하다 console.log(Object.getPrototypeOf(erray)) // 프로토타입의 모습도 확인해보자 // 확인결과 constructor 하나 달랑 있다 Erray.prototype.newone = 123; // 이렇게 값을 넣어준후 console.log(Object.getPrototypeOf(erray)) // 다시 확인해보자 // newone이 들어가있음이 확인된다 console.log(erray.newone) // 그래서 이렇게 접근이 가능하다 console.log(erray.hasOwnProperty('newone')); // 하지만 newone은 erray의 속성인것은 아니라 false가 나온다 // 그런데 잠깐.. erray.hasOwnProperty 는 왜 되는걸까? 생각해보자 console.log(Object.getPrototypeOf(erray)) // 분명 여기에는 newone, constructor만 있을뿐이고 erray의 자체속성에도 hasOwnProperty가 없다 console.log(erray['hasOwnProperty']) // 해보면 나온다 // 자 그 이유는 다음과 같다 console.log(Object.getPrototypeOf(erray)) // 일단 erray의 프로토타입을 확인하고자 한다면 아래와 같이 하면 프로토타입을 볼 수 있다 console.log(Object.getPrototypeOf(Object.getPrototypeOf(erray))) // 그런데 이것의 프로토타입을 또 보자 // 여기에 hasOwnProperty 가 있는것이 확인된다 console.log(Object.getPrototypeOf(Object.getPrototypeOf(Object.getPrototypeOf(erray)))) // 또 씌우면? 이제는 null이 나온다 class Brray { name = 'hello' static type = 'Hello' } let brray = new Brray(); console.log(brray.name) console.log(brray.hasOwnProperty('name')) // 이런 모습의 생성자를 만들어서 객체를 만들어보자 // brray.name 값이 잘 확인된다 // brray.hasOwnProperty 로 name 을 확인해봐도 true로써 속성이 존재함이 확인된다 // static키워드를 붙인 속성인 type은 brray객체와는 전혀 상관없는 요소가 된다. console.log(Brray.type); console.log(Brray.hasOwnProperty('type')); console.log(Object.keys(Brray)); console.log(Object.getOwnPropertyDescriptor(Brray, 'type')) // 이것에 접근하는 방법은 Brray.type이다 // Brray.hasOwnProperty 로 확인했을때 true로써 type은 Brray의 객체에 포함된 속성으로 확인되고 // Object.keys 로도 확인된다. Object.keys로 확인된다는 것은 enumerable이 true라는 뜻이고 Object.getOwnPropertyDescriptor로 정말 그런지 확인해본결과 // 맞는것으로 확인된다 // 즉 static키워드를 붙인것은 Brray.type2 = 'World' // 이런식으로 추가된것과 같다고 볼 수 있다 class Crray extends Array { } let crray = new Crray(); console.log(crray['forEach']); // Array를 상속받는 Crray를 만들어보자 // 그리고 forEach 속성을 확인해본 결과 접근이 가능하다 // 일단 forEach는 crray의 직접적 속성으로 존재하는것은 아니다 console.log(Object.getPrototypeOf(crray) === Crray.prototype) console.log(Object.getPrototypeOf(Object.getPrototypeOf(crray)) === Array.prototype) console.log(Object.getPrototypeOf(Object.getPrototypeOf(Object.getPrototypeOf(crray))) === Object.prototype) // 위와 같은 방법으로 확인해보면 모두 true가 나오는것을 확인할 수 있다 // extends 한다는것은 prototype이 연결되게 됨을 확인할 수 있다 // Array또한 extends Object를 하고있는 모습으로 확인된다 // 그렇기 때문에 Object.prototype 에 있는 모든 함수는 crray 를 통해서도 [] 를 통해서도 접근이 가능한것이다 console.log(crray.hasOwnProperty); console.log([].hasOwnProperty); console.log(crray.hasOwnProperty === Object.prototype.hasOwnProperty); // 이렇게말이다 class Drray { name = 'dd' method() { return 'hello'; } } let drray = new Drray(); console.log(drray.method()) // 위와 같이 method 메소드를 만들어보자 // 그리고 이것은 drray.method() 이런식으로 접근이 가능하다 console.log(drray.hasOwnProperty('method')); console.log(Object.getOwnPropertyNames(drray)) // 이 method란 것이 drray 객체의 속성으로 들어가있을지 확인해봤다. hasOwnProperty로는 false가 나온다 // Object.getOwnPropertyNames 로는 name만 확인된다. // method는 어디에 있는걸까 // 프로토타입을 확인해보자 console.log(Object.getPrototypeOf(drray)) // 여기에 들어가있음이 확인된다 // 즉 Drray.prototype.method2 = function method2() { return 'hello2'; } console.log(drray.method2()) // 해서 만든것과 비슷하다는 의미가 된다. // 이렇게 method2 를 추가한 후 prototype을 확인해보자 console.log(Object.getPrototypeOf(drray)) // 자 이렇게 확인해보자 // method, method2, constructor가 확인되는데 여기서 method2만 enumerable이 true로 보인다. // 정말 그런지 확인해보자 console.log(Object.getOwnPropertyDescriptor(Drray.prototype, 'method')) console.log(Object.getOwnPropertyDescriptor(Drray.prototype, 'method2')) console.log(Object.getOwnPropertyDescriptor(Drray.prototype, 'constructor')) // 확인 결과 method2의 enumerable은 true임이 확인된다 Object.defineProperty(Drray.prototype, 'method3', { value: function method3() { return 'hello3'; }, enumerable: false, configurable: true, writable: true, }); console.log(Object.getOwnPropertyDescriptor(Drray.prototype, 'method')) console.log(Object.getOwnPropertyDescriptor(Drray.prototype, 'method2')) console.log(Object.getOwnPropertyDescriptor(Drray.prototype, 'method3')) console.log(Object.getOwnPropertyDescriptor(Drray.prototype, 'constructor')) // 위와 같이 추가를 하면 모습이 같아지는 것을 확인 할 수 있다 // 참고로 enumerable, configurable, writable 값을 지정을 안해주면 기본은 false이다 // 그러므로 위 상황에서 enumerable는 false로 할것이라면 안써줘도 상관없었다

결론은 같다