자바스크립트 this란 무엇인가
this란 함수를 호출할 때 생성되는 실행 컨텍스트 객체다.
this가 가리키는 대상은 어떻게 this가 호출되는지에 따라 다르다. 아래 상황별로 this의 의미를 나눌 수 있다.
→ 전역 공간에서의 this
→ 메서드로서 호출할 때 내부에서의 this
→ 함수로서 호출할 때 내부에서의 this
→ 콜백 함수 호출 시 내부에서의 this
→ 생성자 함수 내부에서의 this
1. 전역 공간에서의 this
전역 공간에서 this는 전역 객체를 가리킴. (this 콘솔에 찍으면 전역객체 나온다는 뜻)
console.log(this) // {alert: f(), atob: f(), blur: f(), ...}
console.log(window) // {alert: f(), atob: f(), blur: f(), ...}
console.log(this === window) // true
→ this를 콘솔에 찍으나, window를 콘솔에 찍으나 똑같음.
특이점은 우리가 var a = 1; 라고 변수 선언하고 1 할당하면, 자바스크립트 엔진은 전역 객체 안의 프로퍼티로 a를 할당함
var a = 1
console.log(a) // 1
console.log(window.a) // 1
console.log(this.a) // 1
→ a를 직접 콘솔에 찍어도 1이 나오는 건 스코프 체인에서 a를 검색하다가 발견한 프로퍼티이기 때문
→ window.a나 this.a가 1인 건 전역 객체 안의 프로퍼티에 a가 있기 때문
그러나 ES6 이후 var 대신 let과 const를 쓰면 다르게 나온다.
let a = 1
console.log(a) // 1
console.log(window.a) // undefined
console.log(this.a) // undefined
2. 메서드로서 호출할 때 그 메서드 내부에서의 this
일단 함수로서 호출하는 경우와 메서드로서 호출하는 경우를 구분하는 방법 소개
var handleThis = function (x) {
console.log(this, x)
}
handleThis(1) // Window {...} 1
var obj = {
thisMethod: handleThis
}
obj.thisMethod(2) // { thisMethod: f } 2
obj['thisMethod'](2) // { thisMethod: f } 2
→ 호출한 익명함수는 똑같은데, 변수에 담아서 호출한 경우와 객체의 프로퍼티에 할당해서 호출한 경우에 this가 달라짐.
→ 함수로서 호출하는 경우와 메서드로서 호출하는 경우를 구분하는 방법은 함수 앞에 점이 있느냐 없느냐를 확인하는 것. 점 앞에 객체가 명시돼 있기 때문
메서드 내부에서의 this는 어떤 값일까?
var obj = {
methodA: function () {
console.log(this)
},
inner: {
methodB: function () {
console.log(this)
}
}
}
obj.methodA() // { methodA: f, inner: {...} } === obj
obj['methodA']() // { methodA: f, inner: {...} } === obj
obj.inner.methodB() // { methodB: f } === obj.inner
obj.inner['methodB']() // { methodB: f } === obj.inner
obj['inner'].methodB() // { methodB: f } === obj.inner
obj['inner']['methodB']() // { methodB: f } === obj.inner
→ 메서드 내부에서의 this는 호출한 주체에 대한 정보임. 여기서 호출한 주체란 함수명(프로퍼티명) 바로 앞의 객체임. 즉 마지막 점 앞에 명시된 객체를 말함.
3. 함수로서 호출할 때 그 함수 내부에서의 this
그냥 함수로서 호출할 때는 this가 지정되지 않음.
var obj1 = {
outer: function() {
console.log(this) // obj1
var innerFunc = function () {
console.log(this) // Window, obj2
}
innerFunc() // 2번
var obj2 = {
innerMethod: innerFunc
}
obj2.innerMethod() // 3번
}
}
obj1.outer() // 1번
→ 1번: obj1.outer() 함수는 메서드로서 호출한 것. 따라서 outer 함수의 this에는 점 앞의 객체인 obj1이 바인딩됨.
→ 2번: innerFunc()는 객체가 아닌 그냥 변수에 담아서 호출했으므로 this에는 아무것도 지정되지 않아 전역 객체가 콘솔에 찍힘.
→ 3번: obj2.innerMethod()는 위에서 선언한 innerFunc() 함수를 obj2라는 객체의 innerMethod라는 프로퍼티에 할당해서 호출했기 때문에 this에는 점 앞의 객체인 obj2가 바인딩됨.
함수 내부에서 직전 컨텍스트의 this를 바라보게 하는 방법
var obj1 = {
outer: function() {
console.log(this) // { outer: f }
var innerFunc = function () {
console.log(this) // Window
}
innerFunc()
var self = this;
var innerFunc2 = function () {
console.log(self) // { outer: f }
}
innerFunc2()
}
}
obj1.outer()
→ 단순히 self (어떤 변수명이어도 상관 없으나, 컨벤션으로 self라고 쓰는 듯함) 변수에 this를 저장한 후, 함수 안에서 self를 출력해보면 된다
→ 단순히 상위 스코프의 this를 저장해서 내부함수에서 활용하기 위한 수단임
물론 arrow function을 쓰는 지금은 그런 우회는 필요 없다
var obj = {
outer: function () {
console.log(this) // 1. { outer: f }
var innerFunc = () => {
console.log(this) // 2. { outer: f }
}
innerFunc()
}
}
obj.outer()
→ 우회할 필요 없이 함수 안에서 this 찍어 보면 바로 현재 컨텍스트의 this가 나옴
4. 콜백 함수 호출 시 그 함수 내부에서의 this
콜백 함수는 다음 장에서 자세히 다룰 것.
콜백 함수도 기본적으로 함수이기 때문에, 특별히 this를 지정하지 않으면 전역객체를 바라봄. 지정된 경우와 지정되지 않은 경우를 보겠음
// 1번 예시: setTimeout 함수
setTimeout( function () {
console.log(this)
}, 300)
// 2번 예시: forEach 메서드
[1, 2, 3, 4, 5].forEach( function (x) {
console.log(this, x)
})
// 3번 예시: addEventListener
document.body.innerHTML += '<button id="a">클릭</button>'
document.body.querySelector('#a').addEventListener('click', function (e) {
console.log(this, e)
})
// '<div id="root"></div><button id="a">클릭</button>'
→ 1번에서는 내부 콜백 함수에서 따로 this를 지정하지 않으므로 전역객체 출력됨
→ 2번에서도 마찬가지로 내부 콜백 함수에서 this를 지정하지 않았으므로 전역객체와 배열요소 한개씩 5번 출력됨
→ 3번에서는 지정한 객체가 있으므로 점 앞의 엘리먼트와, 이벤트 객체 출력됨
5. 생성자 함수 내부에서의 this
생성자 함수란 어떤 공통된 성질을 지니는 객체들을 생성하는 데 사용하는 함수다.
생성자를 클래스(class), 클래스를 통해 만든 객체를 인스턴스(instance)라고 함
생성자 함수 내부에서의 this는 새로 만들 구체적인 인스턴스 자신이 됨
var Cat = function (name, age) {
this.bark = '애용'
this.name = name
this.age = age
}
var choco = new Cat('초코', 7)
var nabi = new Cat('나비', 5)
console.log(choco, nabi)
// Cat { bark: '애용', name: '초코', age: 7 }
// Cat { bark: '애용', name: '나비', age: 5 }
→ 생성자 함수 Cat을 호출하면, 일단 Cat에 대한 prototype 프로퍼티가 있는 객체가 만들어짐. 그리고 미리 준비된 공통 속성을 해당 객체(this)에 부여함.
→ 그래서 this가 각각 choco 인스턴스, nabi 인스턴스를 가리키는 것.