17장 생성자 함수에 의한 객체 생성
new 연산자와 함께 Object 생성자 함수를 호출하면 빈 객체를 생성해 반환한다. 생성자 함수란 new 연산자와 함께 호출해 객체(인스턴스)를 생성하는 함수이다. 생성자 함수에 의해 생성된 객체를 인스턴스라고 한다. 그러나 Object 생성자 함수를 사용해 객체를 생성하는 방식보다는 객체 리터럴을 사용하는 것이 더 간편하다.
객체 리터럴에 의한 객체 생성 방식은 직관적이고 간편하지만 단 하나의 객체만 생성한다.
객체는 프로퍼티를 통해 객체 고유의 상태를 표현하고 메서드를 통해 프로퍼티를 참조하고 조작하는 동작을 표현하기 때문에, 프로퍼티는 객체마다 프로퍼티 값이 다를 수 있지만 메서드는 내용이 동일한 경우가 일반적이다.
하지만 객체 리터럴에 의해 객체를 생성하는 경우에는 프로퍼티 구조가 동일해도 매번 같은 프로퍼티와 메서드를 기술해야 한다.
생성자 함수에 의한 객체 생성 방식은 마치 객체(인스턴스)를 생성하기 위한 템플릿(클래스)처럼 생성자 함수를 사용해 구조가 동일한 객체 여러 개를 생성할 수 있다.
생성자 함수는 이름 그대로 객체(인스턴스)를 생성하는 함수이지만, 자바같은 클래스 기반 객체지향 언어의 생성자와는 다르게 일반 함수와 동일한 방법으로 생성자 함수를 정의하고 new 연산자와 함께 호출하면 해당 함수는 생성자 함수로 동작한다. new 연산자와 함께 생성자 함수를 호출하지 않으면 일반 함수로 동작한다.
생성자 함수의 역할은 프로퍼티 구조가 동일한 인스턴스를 생성하기 위한 템플릿(클래스)으로서 동작해 인스턴스를 생성하는 것과 생성된 인스턴스를 초기화하는 것이다.
new 연산자와 함께 생성자 함수를 호출하면 자바스크립트 엔진은 암묵적으로 인스턴스를 생성하고 초기화한 후 반환한다.
🧐바인딩 : 식별자와 값을 연결하는 과정. 변수 선언은 변수 이름과 확보된 메모리 공간의 주소를 바인딩하는 것!
[[Call]]
과 [[Construct]]
함수 선언문/함수 표현식으로 정의한 함수는 일반적인 함수로서 호출할 수 있고 생성자 함수로도 호출할 수 있다. 생성자 함수로서 호출한다는 것은 new 연산자와 함께 호출해 객체를 생성하는 것을 말한다.
함수는 객체이지만 일반 객체와 다르다.
일반 객체는 호출할 수 없지만 함수는 호출할 수 있다. 그래서 함수 객체는 일반 객체가 가지고 있는 내부 슬롯과 내부 메서드는 물론, 함수로서 동작하기 위한 [[Environment]]
, [[FormalParameters]]
등의 내부 슬롯과 [[Call]]
, [[Construct]]
같은 내부 메서드를 추가로 가지고 있다.
내부 메서드 [[Call]]
을 갖는 함수 객체를 callable이라 하고, 내부 메서드 [[Construct]]
를 갖는 함수 객체를 constructor, [[Construct]]
를 갖지 않는 함수 객체를 non-constructor라고 한다.
호출할 수 없으면 함수 객체가 아니므로 함수 객체는 반드시 callable이어야 한다. 하지만 모든 함수 객체는 constructo일 수도 있고 non-constructor일 수도 있다.
자바스크립트 엔진은 함수 객체를 생성할 때 함수 정의 방식에 따라 함수를 constructor와 non-constructor로 구분한다.
new 연산자와 함께 호출하는 함수는 non-constructor가 아닌 constructor이어야 한다.
생성자 함수가 new 연산자 없이 호출되는 것을 방지하기 위해 파스칼 케이스 컨벤션을 사용해도 실수는 발생할 수 있기 때문에 ES6에서는 new.target을 지원한다.
new.target은 this와 유사하게 constructor인 모든 함수 내부에서 암묵적인 지역 변수와 같이 사용되고 메타 프로퍼티라고 부른다.
18장 함수와 일급 객체
일급 객체의 조건
자바스크립트의 함수는 위 조건을 모두 만족하는 일급 객체이다.
함수가 일급 객체라는 것은 함수를 객체와 동일하게 사용할 수 있다는 것을 의미한다. 객체는 값이기 때문에 함수는 값과 동일하게 취급할 수 있다. 함수는 값을 사용할 수 있는 곳이라면 어디든지 리터럴로 정의할 수 있고 런타임에 함수 객체로 평가된다.
일급 객체로서 함수가 가지는 가장 큰 특징은 일반 객체와 같이 함수의 매개변수를 전달할 수 있으며, 함수의 반환값으로 사용할 수도 있다는 점이다.
함수 객체는 일반 객체와 달리 호출할 수 있고, 함수 고유의 프로퍼티를 소유한다.
함수의 모든 프로퍼티의 프로퍼티 어트리뷰트를 Object.getOwnPropertyDescriptors
메서드로 확인할 수 있다.
함수 객체의 arguments 프로퍼티 값은 arguments 객체이다. arguments 객체는 함수 호출 시 전달된 인수들의 정보를 담고 있는 유사 배열 객체이고, 함수 내부에서 지역 변수처럼 사용된다.
함수 객체의 arguments 프로퍼티는 ES3부터 폐지되었고, 함수 내부에서 지역 변수처럼 사용할 수 있는 arguments 객체를 참조하도록 한다.
자바스크립트는 함수의 매개변수와 인수의 개수가 일치하는지 확인하지 않아 개수가 달라도 에러가 발생하지 않는다.
함수를 정의할 때 선언한 매개변수는 함수 내부에서 변수와 동일하게 취급된다. 함수가 호출되면 함수 내에서 암묵적으로 매개변수가 선언되고 undefined로 초기화된 후 인수가 할당된다.
선언된 매개변수의 개수보다 인수를 적게 전달하면 인수가 전달되지 않은 매개변수는 undefined로 초기화된 상태를 유지하고, 인수를 더 많이 전달한 경우 초과된 인수는 무시된다. 이때 초과된 인수는 버려지는 것이 아니고 모든 인수는 암묵적으로 arguments 객체의 프로퍼티로 보관된다.
arguments 객체는 인수를 프로퍼티 값으로 소유하고 프로퍼티 키는 인수의 순서를 나타낸다. arguments 객체의 caller 프로퍼티는 호출되어 arguments 객체를 생성한 함수 자신을 가리키고 arguments 객체의 length 프로퍼티는 인수의 개수를 가리킨다.
arguments 객체는 매개변수 개수를 확정할 수 없는 가변 인자 함수를 구현할 때 유용하다.
arguments 객체의 length 프로퍼티는 인자의 개수를 가리키고, 함수 객체의 length 프로퍼티는 매개변수의 개수를 가리킨다.
name 프로퍼티는 ES5와 ES6에서 동작을 다르게 한다. 익명 함수 표현식의 경우 ES5에서의 name 프로퍼티는 빈 문자열을 값으로 갖고,ES6에서는 함수 객체를 가리키는 식별자를 값으로 갖는다.
__proto__
접근자 프로퍼티모든 객체는 [[Prototype]]이라는 내부 슬롯을 갖는다.
[[Prototype]] 내부 슬롯은 객체지향 프로그래밍의 상속을 구현하는 프로토타입 객체를 말한다.
__proto__
프로퍼티는 [[Prototype]] 내부 슬롯이 가리키는 프로토타입 객체에 접근하기 위해 사용되는 접근자 프로퍼티이다.
prototype 프로퍼티는 생성자 함수로 호출할 수 있는 함수 객체, 즉 constructor만이 소유하는 프로퍼티이다.
prototype 프로퍼티는 함수가 객체를 생성하는 생성자 함수로 호출될 때 생성자 함수가 생성할 인스턴스의 프로토타입 객체를 가리킨다.
19장 프로토타입
자바스크립트는 명령형, 함수형, 프로토타입 기반 객체 지향 프로그래밍을 지원하는 멀티 패러다임 프로그래밍 언어이다.
객체지향 프로그래밍은 프로그램을 명령어 또는 함수의 목록으로 보는 전통적인 명령형 프로그래밍의 절차지향적 관점에서 벗어나 객체의 집합으로 프로그램을 표현하려는 프로그래밍 패러다임을 말한다.
객체지향 프로그래밍은 실체(사물/개념)를 인식하려는 철학적 사고를 프로그래밍에 접목하려는 시도에서 시작되는데, 실체는 특징이나 성질을 나타내는 속성을 가지고 있다. 다양한 속성 중에서 프로그래밍에 필요한 속성만 간추려 표현하는 것을 추상화라고 한다.
상속은 객체지향 프로그래밍의 핵심 개념으로, 어떤 객체의 프로퍼티 또는 메서드를 다른 객체가 상속받아 그대로 사용하는 것을 말한다.
동일한 생성자 함수에 의해 생성된 모든 인스턴스가 동일한 메서드를 중복 소유하는 것은 메모리 낭비인데, 자바스크립트는 프로토타입을 기반으로 상속을 구현한다.
상속은 코드의 재사용이라는 점에서 매우 유용하다. 생성자 함수가 생성할 모든 인스턴스가 공통적으로 사용할 프로퍼티나 메서드를 프로토타입에 미리 구현해두면 생성자 함수가 생성할 모든 인스턴스는 상위 객체인 프로토타입의 자산을 공유해 사용할 수 있다.
프로토타입은 어떤 객체의 상위 객체의 역할을 하는 객체로서 다른 객체에 공유 프로퍼티를 제공한다.
모든 객체는 [[Prototype]]
이라는 내부 슬롯을 가지고, 이 내부 슬롯의 값은 프로토타입의 참조이다.
__proto__
접근자 프로퍼티모든 객체는 __proto__
접근자 프로퍼티를 통해 자신의 [[Prototype]]
내부 슬롯에 간접 접근할 수 있다.
함수 객체만이 소유하는 prototype 프로퍼티는 생성자 함수가 생성할 인스턴스의 프로토타입을 가리킨다.
모든 프로토타입은 constructor 프로퍼티를 갖는다. 이 프로퍼티는 prototype 프로퍼티로 자신을 참조하고 있는 생성자 함수를 가리킨다.
리터럴 표기법에 의해 생성된 객체도 프로토타입이 존재하지만 이렇게 생성된 객체의 경우 프로토타입의 constructor 프로퍼티가 가리키는 생성자 함수가 반드시 객체를 생성한 생성자 함수라고 단정할 수 없다.
리터럴 표기법에 의해 생성된 객체도 상속을 위해 프로토타입이 필요하기 때문에 가상적인 생성자 함수를 갖는다. 프로토타입은 생성자 함수와 더불어 생성되고 prototype, constructor 프로퍼티에 의해 연결되어 있다. 즉, 프로토타입과 생성자 함수는 단독으로 존재할 수 없고 언제나 쌍으로 존재한다.
프로토타입은 생성자 함수가 생성되는 시점에 생성된다.
생성자 함수로서 호출할 수 있는 함수, 즉 constructor는 함수 정의가 평가되어 함수 객체를 생성하는 시점에 프로토타입도 생성한다.
빌트인 생성자 함수도 일반 함수와 마찬가지로 빌트인 생성자 함수가 생성되는 시점에 프로토타입이 생성된다.
객체 생성 방법
객체 리터럴에 의해 생성된 obj 객체는 Object.prototype을 프로토타입으로 갖고 Object.prototype을 상속받는다.
Object 생성자 함수를 인수 없이 호출하면 빈 객체가 생성된다. Obejct 생성자 함수를 호출하면 객체 리터럴과 마찬가지로 OrdinaryObjectCreate가 호출된다.
new 연산자와 함께 생성자 함수를 호출해 인스턴스를 생성하면 다른 객체 생성 방식과 마찬가지로 추상 연산 OrdinaryObjectCreate가 호출된다.
자바스크립트는 객체의 프로퍼티에 접근할 때 해당 객체에 접근하려는 프로퍼티가 없다면 [[Prototype]]
내부 슬롯의 참조를 따라 부모 역할 프로토타입의 프로퍼티를 순차적으로 검색하는데, 이것을 프로토타입 체인이라고 한다.
프로토타입 체인의 최상위 객체는 언제나 Object.prototype이다.