콜백 함수는 다른 코드(함수 또는 메서드)에게 인자로 넘겨줌으로써 그 제어권도 위임한 함수다.
콜백 함수의 제어권을 넘겨받은 코드는 콜백 함수 호출 시점에 대한 제어권을 가진다.
예시를 보기 전에 setInterval을 먼저 알아보자.
var intervalID = scope.setInterval(func, delay[, param1, param2, ...]);
//scope는 window or worker의 인스턴스, window생략 가능
//func, dalay 값 필수
//세 번째 매개변수부터 선택적
/*
setInterval 실행하면 반복적 실행되는 내용 자체를 특정할 수 있는 고유ID 반환.
이를 변수에 담는 이유는 반복 실행되는 중간에 종료할 수 있게 하기 위해.
*/
setInterval 콜백 함수의 제어권을 살펴보자.
var count = 0;
var cbFunc = function () {
console.log(count);
if(++count > 4) clearInterval(timer);
};
var timer = setInterval(cbFunc, 300);
코드를 실행하면 0.3초에 한 번씩 숫자가 0부터 1씩 증가하다 4가 출력된 이후 종료된다. cbFunc가 아닌 setInterval에 cbFunc 함수를 넘겼고, setInterval은 콜백 함수 호출 시점에 대한 제어권을 갖게 되었다.
콜백 함수를 호출할 때 인자로 넘겨줄 값들 및 순서는 정해져있다. 반드시 이 순서를 따라야 한다.
콜백 함수도 함수니까 this는 전역객체를 참조하지만, 제어권을 넘겨받을 코드에서 콜백 함수에 this가 될 대상을 지정한 경우 그 대상을 참조한다.
콜백 함수로 어떤 함수의 인자에 객체의 메서드를 전달해도 그 메서드는 함수로서 호출된다.
var obj = {
vals: [1, 2, 3],
logvalues: function(v,i) {
console.log(this, v, i);
}
};
obj.logvalues(1, 2); //메서드로 호출, this는 obj를 가리킴
[4, 5, 6].forEach(obj.logValues); //this는 전역객체
// logValues메서드를 forEach 함수의 콜백 함수로 전달.
// obj.logvalues가 가리키는 함수만 전달한 것임.
// forEach에 의해 콜백이 함수로 호출된다.
위와 같이, 메서드를 콜백 함수로 전달하면 this는 전역객체를 바라본다. this가 객체를 바라보게 하고 싶다면 어떻게 해야할까? 전통적으로는 this를 다른 변수에 담아 콜백 함수로 활용했다. 이는 너무 번거롭고, 이를 보완하는 방식으로 ES5에서 등장한 bind 메서드를 이용할 수 있다.
var obj1 = {
name: 'obj1',
func: function () {
console.log(this.name);
}
};
setTimeout(obj1.func.bind(obj1), 1000); //"obj1"
아래 이미지는 콜백 지옥 예시다,,ヽ( ̄д ̄;)ノ (이미지 출처)
지금 당장 김밥이 먹고 싶어 김밥리스트로 콜백 지옥을 만들었다.
//callback hell
setTimeout(function (name) {
var gimbapList = name;
console.log(gimbapList);
setTimeout(function (name) {
gimbapList += ',' + name;
console.log(gimbapList);
setTimeout(function (name) {
gimbapList += ',' + name;
console.log(gimbapList);
setTimeout(function (name) {
gimbapList += ',' + name;
console.log(gimbapList);
}, 500, '치즈김밥');
}, 500, '참치김밥');
}, 500, '떡갈비김밥');
}, 500, '야채김밥');
/*
"야채김밥"
"야채김밥,떡갈비김밥"
"야채김밥,떡갈비김밥,참치김밥"
"야채김밥,떡갈비김밥,참치김밥,치즈김밥"
*/
가독성도 떨어지지만, 값이 전달되는 순서가 아래에서 위로 향해 어색하게 느껴지기도 한다. 이런 문제를 해결하기 위해 여러 가지 방법을 사용할 수 있다.
김밥리스트 콜백 지옥을 해결해보자.
var gimbapList = '';
var add1 = function(name) {
gimbapList = name;
console.log(gimbapList);
setTimeout(add2, 500, '떡갈비김밥')
};
var add2 = function(name) {
gimbapList += ',' + name;
console.log(gimbapList);
setTimeout(add3, 500, '참치김밥')
};
var add3 = function(name) {
gimbapList += ',' + name;
console.log(gimbapList);
setTimeout(add4, 500, '치즈김밥')
};
var add4 = function(name) {
gimbapList += ',' + name;
console.log(gimbapList);
};
setTimeout(add1, 500, '야채김밥')
/*
"야채김밥"
"야채김밥,떡갈비김밥"
"야채김밥,떡갈비김밥,참치김밥"
"야채김밥,떡갈비김밥,참치김밥,치즈김밥"
*/
자바스크립트 진영은 비동기적인 작업을 동기적으로, 혹은 동기적인 것처럼 보이게 처리해주는 장치를 마련하고자 노력하고 있다. ES6에서는 Promise, Generator 등이 도입됐고, ES2017에서는 async/await가 도입됐다.
https://developer.mozilla.org/ko/docs/Web/JavaScript/Reference/Global_Objects/Promise
promise는 비동기 작업이 맞이할 미래의 완료 또는 실패와 그 결과 값을 나타낸다. 비동기 작업이 완료될 때 resolve/reject를 호출하는 방법으로 동기적 표현이 가능하며, 둘 중 하나가 실행되기 전까지는 다음 또는 오류 구문으로 넘어가지 않는다.
new Promise(function (resolve) {
setTimeout(function () {
var name = '야채김밥';
console.log(name);
resolve(name);
}, 500);
}).then(function (prevName) {
return new Promise(function (resolve) {
setTimeout(function () {
var name = '떡갈비김밥';
console.log(name);
resolve(name);
}, 500);
});
}).then(function (prevName) {
return new Promise(function (resolve) {
setTimeout(function () {
var name = '참치김밥';
console.log(name);
resolve(name);
}, 500);
});
}).then(function (prevName) {
return new Promise(function (resolve) {
setTimeout(function () {
var name = '치즈김밥';
console.log(name);
resolve(name);
}, 500);
});
});
/*
"야채김밥"
"떡갈비김밥"
"참치김밥"
"치즈김밥"
*/
generator함수를 실행하면 iterator를 반환한다. iterator는 next메서드를 가지는데, 이 메서드를 호출하면 generator함수 내부에서 가장 먼저 등장하는 yield에서의 함수 실행을 멈춘다.
var addGimbap = function (prevName, name) {
setTimeout(function () {
gimbapMaker.next(prevName ? prevName + ',' + name : name);
}, 500);
};
var gimbapGenerator = function* () {
var vegGimbap = yield addGimbap('', '야채김밥');
console.log(vegGimbap);
var galbiGimbap = yield addGimbap('', '떡갈비김밥');
console.log(galbiGimbap);
var tunaGimbap = yield addGimbap('', '참치김밥');
console.log(tunaGimbap);
var cheeseGimbap = yield addGimbap('', '치즈김밥');
console.log(cheeseGimbap);
};
var gimbapMaker = gimbapGenerator();
gimbapMaker.next();
/*
"야채김밥"
"떡갈비김밥"
"참치김밥"
"치즈김밥"
*/
이 작업으로 뒤의 내용을 promise로 자동 전환할 수 있다. 즉, promise의 then같은 효과를 얻을 수 있다.
var addGimbap = function (name) {
return new Promise(function (resolve) {
setTimeout(function () {
resolve(name);
}, 500);
});
};
var gimbapMaker = async function () {
var gimbapList = '';
var _addGimbap = async function (name) {
gimbapList += (gimbapList ? ',' : '') + await addGimbap(name);
};
await _addGimbap('야채김밥');
console.log(gimbapList);
await _addGimbap('떡갈비김밥');
console.log(gimbapList);
await _addGimbap('참치김밥');
console.log(gimbapList);
await _addGimbap('치즈김밥');
console.log(gimbapList);
};
gimbapMaker();
/*
"야채김밥"
"야채김밥,떡갈비김밥"
"야채김밥,떡갈비김밥,참치김밥"
"야채김밥,떡갈비김밥,참치김밥,치즈김밥"
*/
//오타때문에 한참을 헤맸다 ㅠ-ㅠ 소괄호 대괄호 잘 닫아주고, 변수명은 쉽게...!!!