반응형

자바스크립트 this를 정복하는 포스팅입니다. 모던 자바스크립트 deep dive 22장 this 내용을 적은 글입니다. this 개념은 배울 때마다 헷갈리는 개념이라 마음잡고 정리했습니다. 

자바스크립트 this 정복하기

(아래 사진은 this를 바라보는 저의 모습과 비슷해서 가져왔습니다.두려움, 어려움, 설명할 수 없는 개념...)

Javascript This 정복하기

this가 필요한 이유

아래와 같은 상황일 때 this를 쓸 수 있다. 

const circle = {
  radius:5,
  getDiameter(){
    //이 메서드가 자신이 속한 객체의 프로퍼티나 다른 메서드를 참조하려면 
    //자신이 속한 객체인 circle을 참조할 수 있어야 한다. 
    return 2*circle.radius;
  }
};

console.log(circle.getDiameter());//10

getDiameter 메서드 내에서 메서드 자신이 속한 객체를 가리키는 식별자 circle을 참조하고 있다. getDiameter가 호출되는 시점에는 이미 객체 리터럴의 평가가 완료되어 객체가 생성되었고, circle 식별자에 생성된 객체가 할당됐다. 하지만 자기 자신이 속한 객체를 재귀적으로 참조하는 방식은 일반적이지 않고, 바람직하지 않다. 그래서 this가 필요하다. 

const circle = {
  radius:5,
  getDiameter(){
    //this는 메서드를 호출한 객체를 가리킨다.  
    return 2*this.radius;
  }
};

console.log(circle.getDiameter());//10

this는 자신이 속한 객체 또는 자신이 생성할 인스턴스를 가리키는 자기 참조 변수(self-referencing variable)다. this를 통해 자신이 속한 객체 또는 자신이 생성할 인스턴스의 프로퍼티나 메서드를 참조할 수 있다.


this 바인딩

바인딩이란 식별자연결하는 과정을 의미한다. 예를 들어, 변수 선언은 변수 이름(식별자)확보된 메모리 공간의 주소를 바인딩하는 것이다. this 바인딩은 this(식별자 역할)와 this가 가리킬 객체바인딩하는 것이다. 

function Circle(radius){
  //this는 생성자 함수가 생성할 인스턴스를 가리킨다.
  this.radius = radius;
}

Circle.prototype.getDiameter = function(){
  //this는 생성자 함수가 생성할 인스턴스를 가리킨다.
  return 2*this.radius;
};

//인스턴스 생성
const circle = new Circle(5);
console.log(circle.getDiameter());//10

생성자 함수 내부의 this는 생성자 함수가 생성할 인스턴스를 가리킨다. 


this 바인딩은 함수가 어떻게 호출되었는지에 따라 동적으로 결정된다. 

일반 함수 호출 this 바인딩

일반적으로 this에는 전역 객체 (global object)가 바인딩된다. 

function foo(){
  console.log('foo this', this); //window
}

foo();

strict mode에서는 전역 객체 대신 undefined가 바인딩된다. 

*this개념: 자신이 속한 객체 또는 자신이 생성할 인스턴스를 가리키는 자기 참조 변수다. this를 통해 자신이 속한 객체 또는 자신이 생성할 인스턴스의 프로퍼티나 메서드를 참조할 수 있다. 

=> 객체를 생성하지 않는 일반 함수(중첩 함수, 콜백 함수 포함)에서 this는 의미가 없다.


콜백 함수 내부 전역 객체 바인딩 막기

var value = 1;

const obj = {
  value: 100,
  foo() {
    //콜백 함수여서 전역 객체 바인딩된다. 
    setTimeout(function () {
      console.log("callback this:", this); //window
    }, 100);
  },
};

obj.foo();

개선 코드

const that = this;를 추가하여 this 바인딩을 변수 that에 할당해 준다. this (전역 객체) 대신에 that을 참조한다. 

또는 명시적으로 this를 바인딩해 준다. 

var value = 1;

const obj = {
  value: 100,
  foo() {
    const that = this;
    setTimeout(function () {
      console.log("callback this:", that.value); //100
    }, 100);
  },
};

obj.foo();
    //bind를 사용하여 명시적으로 this를 바인딩하기.
    foo(){
    setTimeout(function(){
      console.log(this.value);//100
    }.bind(this),100);
    }

메서드 호출 this 바인딩

메서드 이름 앞의 마침표(.) 연산자 앞에 기술한 객체가 바인딩된다. 메서드를 호출한 객체에 바인딩된다. 

const Person = {
  name :'Lee',
  getName(){
    //메서드 내부의 this는 메서드를 호출한 객체에 바인딩된다. 
    return this.name;
  }
};

//person 객체에 연결된다.
console.log(person.getName()); //Lee

생성자 함수 호출 this 바인딩

생성자 함수 내부의 this에는 생성자 함수가 (미래에) 생성할 인스턴스가 바인딩된다. 

function Circle(radius){
  //생성자 함수 내부의 this는 생성자 함수가 생성할 인스턴스를 가리킨다. 
  this.radius = radius;
  this.getDiameter = function(){
    return 2*this.radius;
  };
}

//반지름이 5인 Circle 객체를 생성
const circle1 = new Circle(5);
//생성자 함수가 생성한 인스턴스를 가리킨다. 
console.log(circle1.getDiameter());//10

bind 메서드

Function.prototype.bind 메서드는 apply와 call 메서드와 달리 함수를 호출하지 않는다. 다만 첫 번째 인수로 전달한 값으로 this 바인딩이 교체된 함수를 새롭게 생성해 반환한다. 

const person = {
  name :'Lee',
  foo(callback){
    //bind 메서드로 callback 함수 내부의 this 바인딩을 전달
    setTimeout(callback.bind(this),100);
    //setTimeout(callback,100);
    //bind를 안쓰면 전역객체 window를 가리킨다. 
  }
};

person.foo(function(){
  console.log(`Hi my name is ${this.name}.`);
});
반응형