22-1. this 키워드

동작을 나타내는 메서드는 자신이 속한 객체의 상태인 프로퍼티를 참조하고 변경하려면,

자신이 속한 객체를 가리키는 식별자를 참조할 수 있어야 한다.

[객체 리터럴]

: 객체 리터럴로 생성된 객체에서는 식별자를 자기 참조, 재귀적으로 참조 가능함

왜냐하면, 변수에 객체가 할당되기 직전에 객체 리터럴이 평가되기 때문에

식별자에 객체가 할당이 되어 있어서

메서드에서 식별자를 자기 참조할때에는 참조가 가능한 것

22-01

const circle = { // 객체 리터럴은 circle 변수에 할당되기 직전에 평가됨
	// 프로퍼티 : 객체 고유의 상태 데이터
	 radius: 5,
	// 메서드: 상태 데이터를 참조하고 조작하는 동작
	getDiameter() {
		// 이 메서드가 자신이 속한 객체의 프로퍼티나 다른 메서드를 참조하려면, 
		// 자신이 속한 객체인 circle을 참조할 수 있어야 한다.
		return 2 * circle.radius;
	}
}
cosnole.log(cirlce.getDiameter()); // 10{}

[생성자 함수]

: 생성자 함수가 정의된 이후

new 연산자와 함께 생성자 함수가 호출이 되어야 인스턴스가 생성이 되므로

생성자 함수 안에서는 아직 생성되기 전인 인스턴스 식별자를 알수가 없다.

그래서 특수한 식별자가 필요하고, 그게 this 임

22-02

function Circle (radius) {
	// 이 시점에는 생성자 함수 자신이 생성할 인스턴스를 가리키는 식별자를 알 수 없다.
	????.radius = radius;
}

Circle.prototype.getDiameter = function () {
	// 이 시점에는 생성자 함수 자신이 생성할 인스턴스를 가리키는 식별자를 알 수 없다.
	return 2 * ????.radius; 
}

// 생성자 함수로 인스턴스를 생성하려면 먼저 생성자 함수를 정의해야 한다.
const circle = new Circle(5);

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

this는 자바스크립트 엔진에 의해 암묵적으로 생성, 어디서든 참조 가능

this가 가리키는 값, 즉 this 바인딩은 함수 호출 방식에 의해 동적으로 결정된다.

[객체 리터럴]

22-03

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

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

⇒ 여기서 this는 circle 객체를 가리킴

[생성자 함수]

22-04

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

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

// 인스턴스 생성
const circle = new Circle(5);

⇒ this는 인스턴스를 가리킴

22-05

// this는 어디서든지 참조 가능
// 전역에서 this는 전역 객체 window를 가리킨다.
console.log(this); // window

function square (number) {
	console.log(this); // window
	return number * number;
}
square(2);

---------------------------------------------------------------------

const person = {
	name: 'Lee',
	getName () {
		// 메서드 내부에서 this는 메서드를 호출한 객체를 가리킨다.
		console.log(this); // {name: 'Lee', getName: f}
		reuturn this.name;
	}
}

---------------------------------------------------------------------

function Person (name) {
	this.name = name;
	// 생성자 함수 내부에서 this는 생성자 함수가 생성할 인스턴스를 가리킨다.
	console.log(this); // Person {name: 'Lee'}
}

const me = new Person('Lee');

strict mode가 적용된 일반 함수 내부의 this에는 undefined가 바인딩된다.

22-2. 함수 호출 방식과 this 바인딩

this 바인딩은

함수 호출 방식, 즉 함수가 어떻게 호출되었는지에 따라 동적으로 결정

[함수 호출하는 방식]

22-06

// this 바인딩은 함수 호출 방식에 따라 동적으로 결정된다.
const foo = function () {
	console.dir(this); 
};

// 동일한 함수도 다양한 방식으로 호출할 수 있다.

---------------------------------------------------------------------

// 1. 일반 함수 호출
// foo 함수를 일반적인 방식으로 호출
// foo 함수 내부의 this는 전역 객체 window를 가리킨다.
foo(); // window

// 2. 메서드 호출
// foo 함수를 프로퍼티 값으로 할당하여 호출
// foo 함수 내부의 this는 메서드를 호출한 객체 obj를 가리킨다.
const obj = {foo};
obj.foo(); // obj

// 3. 생성자 함수 호출
// foo 함수를 new 연산자와 함께 생성자 함수로 호출
// foo 함수 내부의 this는 생성자 함수가 호출할 인스턴스를 가리킨다.
new foo (); // foo {}

// 4. Function.prototype.apply/call/bind 메서드에 의하 간접 호출
// foo 함수 내부의 this는 인수에 의해 결정된다.
const bar = {name: 'bar'};

foo.call(bar); // bar
foo.apply(bar); // bar
foo.bind(bar); // bar

  1. 일반 함수 호출

: 일반 함수로 호출된 모든 함수 내부의 this는 전역 객체에 바인딩

22-07

function foo () {
	console.log("foo's this: ", this); // window
	function bar () {
		console.log("bar's this: ", this); // window
	}
	bar();
}
foo();

---------------------------------------------------------------------

22-08 - strict mode 일때 undefined

function foo () {
	'use strict';
	console.log("foo's this: ", this); // foo's this:  undefined
	function bar () {
		console.log("bar's this: ", this); // bar's this:  undefined
	}
	bar();
}
foo();

---------------------------------------------------------------------

22-09 - 메서드 내의 중첩 함수 일때

// var 키워드로 선언한 전역 변수 value는 전역 객체의 프로퍼티다.
var value = 1;

// const 키워드로 선언한 전역 변수 value는 전역 객체의 프로퍼티가 아니다.
const obj = {
	value: 100,
	foo () {
		console.log("foo's this: ", this); // {valuse: 100, foo: f}
		console.log("foo's this.value: ", this.value); // 100
		
		// 메서드 내에서 정의한 중첩 함수
		function bar () {
			console.log("bar's this: ", this); // window
			console.log("bar's this.value: ", this.value); // 1
		}
		// 메서드 내에서 정의한 중첩 함수도 일반 함수로 호출되면 
		// 중첩 함수 내부의 this는 전역 객체가 바인딩 된다.
		bar();
	}
}

---------------------------------------------------------------------

22-10 - 메서드 내의 콜백 함수일때 

var value = 1;

const obj = {
	value: 100,
	foo () {
		console.log("foo's this: ", this); // {value: 100, foo: f}
		// 콜백 함수 내부의 this는 전역 객체가 바인딩된다.
		setTimeout(function () {
			console.log("callback's this: ", this); // window
			console.log("callback's this.value: ", this); // 1
		})
	}
}

⇒ 그런데,

일반 함수로 호출을 할때

메서드 내의 중첩 함수 또는 콜백함수는 외부함수를 돕는 역할을 하기도 하는데,

외부 함수인 메서드는 자기 참조를 하지만,

메서드 내 중첩 함수 또는 콜백함수는 전역객체를 참조하면 헬퍼 함수로 동작하기 어려움

22-11 예제처럼 this를 변수값에 할당해서 해결할 수 있음!

22-11

var value = 1;

const obj = {
	value: 100,
	foo () {
		// this 바인딩(obj)을 변수 that에 할당한다.
		const that = this;
		
		// 콜백 함수 내부에서 this 대신 that을 참조한다.
		setTimeout(function () {
			console.log(that.value); // 100
		}, 100);
	}
}

또는

22-12 예제처럼 Function.prototype.apply, Function.prototype.call, Function.prototype.bind으로 해결할수 있음!

22-12

var value = 1;

const obj = {
	value: 100,
	foo() {
		// 콜백 함수에 명시적으로 this를 바인딩한다.
		setTimeout(function () {
			console.log(this.value); // 100
		}.bind(this), 100);
	}
}

또는

22-13 예제처럼 화살표 함수로 해결할 수 있음

22-13

var value = 1;

const obj = {
	value: 100,
	foo () {
		// 화살표 함수 내부의 this는 상위 스코프의 this를 가리킨다.
		setTimeout(() => console.log(this.value), 100); // 100
	}
}
  1. 메서드 호출

: 메서드 내부의 this는

메서드를 소유한 객체가 아닌 메서드를 호출한 객체에 바인딩!

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

// 메서드는 getName을 호출한 객체는 person이다.
console.log(person.getName()); // Lee

---------------------------------------------------------------------

22-15

const anotherPerson = {
	name: 'Kim'
};

// getName 메서드를 anotherPerson 객체의 메서드로 할당
anotherPerson.getName = person.getName;

// getName 메서드를 호출한 객체는 anotherPerson이다.
console.log(anotherPerson.getName()); // Kim

// getName 메서드를 변수에 할당
const getName = person.getName;

// getName 메서드를 일반 함수로 호출
console.log(getName()); // ''
// 일반 함수로 호출된 getName 함수 내부의 this.name은 브라우저 환경에서 window.name과 같다
// 브라우저 환경에서 window.name은 브라우저 창의 이름을 나타내는 빌트인 프로퍼티이며 기본값으 ''
// Node.js 환경에서 this.name은 undefined이다

---------------------------------------------------------------------

22-16

function Person (name) {
	this.name = name;
}

Person.prototype.getName = function () {
	return this.name;
}

const me = new Person ('Lee');

// getName 메서드를 호출한 객체는 me다.
console.log(me.getName()); // Lee

Person.prototype.name = 'Kim';

// getName 메서드를 호출한 객체는 Person.prototype이다.
console.log(Person.prototype.getName()); // Kim
  1. 생성자 함수 호출

: 생성자 함수 내부의 this는

생성자 함수가 미래에 생성할 인스턴스가 바인딩

22-17

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

// 반지름이 5인 Circle 객체를 생성
const circle1 = new Circle(5);
// 반지름이 10인 Circle 객체를 생성
const circle2 = new Circle(10);

console.log(circle1.getDiameter()); // 10
console.log(circle2.getDiameter()); // 20
  1. Function.prototype.apply/call/bind 메서드에 의한 간접 호출

apply

call

bind

는 Function.prototype의 메서드임

Function.prototype.apply,

Function.prototype.call 메서드는 this로 사용할 “객체”와 “인수 리스트”를 인수로 전달받아 함수로 호출

[사용법]

thisArg : this로 사용할 객체
argsArray : 함수에게 전달할 인수 리스트의 배열 또는 유사 배열 객체

Function.prototype.apply(thisArg[, argsArray])

thisArg : this로 사용할 객체
arg1, arg2, ... : 함수에게 전달할 인수 리스트
Function.prototype.call(thisArg[, arg1[, arg2[, ...]]])

apply와 call 메서드의 본질적인 기능은 함수를 호출하는 것
22-19

function getThisBinding () {
	return this;
}

// this로 사용할 객체
const thisArg = {a: 1};

console.log(getThisBinding()); // window

// getThisBinding 함수를 호출하면서 인수로 전달한 객체를 getThisBinding 함수의 this에 바인딩한다.
console.log(getThisBinding.apply(thisArg)); // {a: 1}
console.log(getThisBinding.call(thisArg)); // {a: 1}

--------------------------------------------------------------------------

22-20

function getThisBinding () {
	console.log(arguments);
	return this;
}

// this로 사용할 객체
const thisArg = {a: 1};

// 1) getThisBinding 함수를 호출하면서 인수로 전달한 객체를 getThisBinding 함수의 this에 바인딩한다.
// 2) apply 메서드는 호출할 함수의 인수를 배열로 묶어 전달한다.
console.log(getThisBinding.apply(thisArg, [1,2,3]));
// 1)의 결과 : Arguments(3) [1, 2, 3, callee: ƒ, Symbol(Symbol.iterator): ƒ]
// 2)의 결과 : {a: 1}

// 3) call 메서드는 호출할 함수의 인수를 쉼표로 구분한 리스트 형식으로 전달한다.
console.log(getThisBinding.call(thisArg, 1, 2, 3));
// 1)의 결과 : Arguments(3) [1, 2, 3, callee: ƒ, Symbol(Symbol.iterator): ƒ]
// 2)의 결과 : {a: 1}

call 과 apply 사용 목적

: arguments와 같은 유사 배열 객체에 배열 메서드를 사용하려고

22-21

function convertArgsToArray () {
	console.log(arguments);
	
	// arguments 객체를 배열로 변환
	// Array.prototype.slice를 인수 없이 호출하면 배열의 복사본을 생성한다.
	const arr = Array.prototype.slice.call(arguments);
	// const arr = Array.prototype.slice.apply(arguments);
	console.log(arr);

	return arr;
}

convertArgsToArray(1,2,3); // [1,2,3]

⇒ arguments 객체는 배열이 아니기 때문에

Array.prototype.slice 배열의 메서드를 사용할 수 없지만

apply, call 메서드를 뒤에 붙여서 사용하면 쓸수있음

22-22

function getThisBinding () {
	return this;
}

// this로 사용할 객체
const thisArg = {a : 1};

// bind 메서드는 첫 번째 인수로 전달한 thisArg로 this 바인딩이 교체된
// getThisBinding 함수를 새롭게 생성해 반환한다.
console.log(getThisBinding.bind(thisArg)); // getThisBinding
// bind 메서드는 함수를 호출하지 않으므로 명시적으로 호출해야 한다.
console.log(getThisBinding.bind(thisArg)()); // {a: 1}

⇒ bind 메서드는 this 바인딩이 교체된 함수를 새롭게 생성해 반환함

22-23

const person = {
	name: 'Lee',
	foo(callback) {
		setTimeout(callback, 100);
	}
}

person.foo(function () {
	console.log(`Hi! my name is ${this.name}.`); //Hi! my name is .
	// 일반 함수로 호출된 콜백 함수 내부의 this.name은 브라우저 환경에서 window.name과 같다.
	// 브라우저 환경에서 window.name은 브라우저 창의 이름을 나타내는 빌트인 프로퍼티이며 기본값은 ''다.
	// Node.js 환경에서 this.name은 undefined이다.
})

⇒ 콜백함수가 헬퍼함수로 동작하여 같은 this가 바인딩되도록 하려면 bind 메서드를 사용하면 됨

22-24

const person = {
	name: 'Lee',
	foo(callback) {
		// bind 메서드로 callback 함수 내부의 this 바인딩을 전달
		setTimeout(callback.bind(this), 100);
	}
}

person.foo(function () {
	console.log(`Hi! my name is ${this.name}.`); // Hi! my name is Lee.
})