Deep Dive 22. this
this 키워드
메서드가 자신이 속한 객체의 프로퍼티를 참조하려면 먼저 자신이 속한 객체를 가리키는 식별자를 참조할 수 있어야 한다.
1
2
3
4
5
6
const circle = {
radius: 5,
getDiameter(){
return 2 * circle.radius;
}
};
객체 리터럴로 객체를 생성하는 경우에는 getDiameter를 정의하는 과정에서 circle이 상위 스코프에 있기 때문에 this를 사용하지 않고 circle을 가리킬 수 있다. 하지만 자기 자신이 속한 객체를 재귀적으로 참조하는 방식은 일반적이지 않고 바람직하지도 않다.
1
2
3
4
5
6
7
8
function Circle(radius){
???.radius = radius; // 이 시점에서 생성자 함수 자신이 생성할 인스턴스를 가리키는 식별자를 알 수 없다.
}
Circle.prototype.getDiameter = function (){
return 2 * ????.radius; // 이 시점에서 생성자 함수 자신이 생성할 인스턴스를 가리키는 식별자를 알 수 없다.
};
const circle = new Circle(5);
생성자 함수를 정의하는 시점에는 아직 인스턴스를 생성하기 이전이므로 생성자 함수가 생성할 인스턴스를 가리키는 식별자를 알 수 없다. 따라서 자신이 속한 객체 또는 자신이 생성할 인스턴스를 가리키는 특수한 식별자 this
가 필요하다.
this
는 자식이 속한 객체 또는 자신이 생성할 인스턴스를 가리키는 자기 참조 변수다. this
를 통해 자신이 속한 객체 또는 자신이 생성할 인스턴스의 프로퍼티나 메서드를 참조할 수 있다.
this
는 자바스크립트 엔진에 의해 암묵적으로 생성되며, 코드 어디서든 참조할 수 있다. 함수를 호출하면 arguments
객체와 this
가 암묵적으로 함수 내부에 전달된다.
1
2
3
4
5
6
7
8
function Circle(radius){
this.radius = radius; // this는 생성자 함수가 생성할 인스턴스를 가리킨다.
}
Circle.prototype.getDiameter = function (){
return 2 * this.radius; // this는 생성자 함수가 생성할 인스턴스를 가리킨다.
}
const circle = new Circle(5);
자바나 C++ 같은 클래스 기반 언어에서 this는 언제나 클래스가 생성하는 인스턴스를 가리킨다. 하지만 자바스크립트의 this
바인딩은 함수 호출 방식에 의해 동적으로 결정된다.
함수 호출 방식과 this 바인딩
일반 함수 호출
기본적으로 this에는 전역 객체가 바인딩 된다.
1
2
3
4
5
6
7
8
function foo(){
console.log("foo's this: ",this); // window
function bar(){
console.log("bar's this: ", this); // window
}
bar();
}
foo();
일반 함수로 호출된 모든 함수(중첨 함수, 콜백 함수 포함) 내부의 this에는 전역 객체가 바인딩된다. 전역 함수는 물론이고 중첩 함수를 일반함수로 호출하면 함수 내부의 this에는 전역 객체가 바인딩된다. 일반적으로 this는 객체의 프로퍼티나 메서드를 참조하기 위한 자기 참조 변수이므로 객체를 생성하지 않는 일반 함수에서 this는 의미가 없다.
메서드 호출
메서드 내부의 this에는 메서드를 호출한 객체, 즉 메서드를 호출할 때 메서드 이름 앞의 마침표(.) 연산자 앞에 기술한 객체가 바인딩된다. 주의할 것은 메서드 내부의 this는 메서드를 소유한 객체가 아닌 메서드를 호출한 객체에 바인딩된다는 것이다.
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
const person = {
name: 'Lee',
getName(){
// 메서드 내부의 this는 메서드를 호출한 객체에 바인딩된다.
return this.name;
}
};
// 메서드 getName을 호출한 객체는 person이다.
console.log(person.getName()); // Lee
const anotherPerson = {
name: 'Kim'
};
anotherPerson.getName = person.getName;
console.log(anotherPerson.getName()); // Kim
생성자 함수 호출
생성자 함수 내부의 this에는 생성자 함수가 미래에 생성할 인스턴스가 바인딩된다.
1
2
3
4
5
6
7
8
9
10
function Circle(radius){
this.radius = radius;
this.getDiameter = function (){
return 2 * this.radius;
};
}
const circle = new Circle(5);
console.log(circle.getDiameter()); // 10;