클로저란?
MDN에서는 클로저를 다음과 같이 정의하고 있다.
클로저는 독립적인 (자유) 변수를 가리키는 함수이다. 또는, 클로저 안에 정의된 함수는 만들어진 환경을 ‘기억한다’.
외부 함수가 반환된 후에도 외부 함수의 변수 범위 체인에 접근할 수 있는 함수이다. 전역 변수의 사용을 억제하고, 정보를 은닉하기 위해 사용한다.
흔히 함수 내에서 함수를 정의하고 사용하면 클로저라고 한다.
하지만 대개는 정의한 함수를 리턴하고 사용은 바깥에서 하게 된다.
function getClosure() {
var text = 'variable 1';
return function() {
return text;
};
}
var closure = getClosure();
console.log(closure()); // 'variable 1'
위에서 정의한 getClosure() 는 함수를 반환하고, 반환된 함수는 getClosure() 내부에서 선언된 변수를 참조하고 있다.
또한 이렇게 참조된 변수는 함수 실행이 끝났다고 해서 사라지지 않았고,
여전히 제대로 된 값을 반환하고 있는 것을 알 수 있다.
여기서 반환된 함수가 클로저인데, MDN에서 정의된 내용에서도 말했듯 환경을 기억하고 있는 것처럼 보인다.
클로저를 통한 은닉화
일반적으로 JavaScript에서 객체지향 프로그래밍을 말한다면 Prototype을 통해 객체를 다루는 것을 말한다.
Prototype을 통한 객체를 만들 때의 주요한 문제 중 하나는 Private variables에 대한 접근 권한 문제이다.
function Hello(name) {
this._name = name;
}
Hello.prototype.say = function() {
console.log('Hello, ' + this._name);
}
var hello1 = new Hello('승민');
var hello2 = new Hello('현섭');
var hello3 = new Hello('유근');
hello1.say(); // 'Hello, 승민'
hello2.say(); // 'Hello, 현섭'
hello3.say(); // 'Hello, 유근'
hello1._name = 'anonymous';
hello1.say(); // 'Hello, anonymous'
위에서 Hello()로 생성된 객체들은 모두 _name 이라는 변수를 가지게 된다.
변수명 앞에 underscore(_)를 포함했기 때문에 일반적인 JavaScript 네이밍 컨벤션을 생각해 봤을 때 이 변수는 Private variable으로 쓰고 싶다는 의도를 알 수 있다. 하지만 실제로는 여전히 외부에서도 쉽게 접근가능한 변수일 뿐이다.
이 겨우에 클로저를 사용하여 외부에서 변수에 직접 접근한 것을 제한할 수 있다.
function hello(name) {
var _name = name;
return function() {
console.log('Hello, ' + _name);
};
}
var hello1 = hello('승민');
var hello2 = hello('현섭');
var hello3 = hello('유근');
hello1(); // 'Hello, 승민'
hello2(); // 'Hello, 현섭'
hello3(); // 'Hello, 유근'
특별히 인터페이스를 제공하는게 아니라면 여기서는 외부에서 _name에 접근할 방법이 전혀 없다.
반복문 클로저
var i;
for (i = 0; i < 10; i++) {
setTimeout(function() {
console.log(i);
}, 100);
}
간단하게 0-9까지의 정수를 출력하는 코드이지만 실제로는 엉뚱하게도 10만 열 번 호출되는 것을 볼 수 있다.
왜일까?
먼저 setTimeout()에 인자로 넘긴 익명함수는 모두 0.1초 뒤에 호출될 것이다.
그 0.1초 동안에 이미 반복문이 모두 순회하면서 i 값은 이미 10이 된 상태.
그 때 익명함수가 호출되면서 이미 10이 되어버린 i를 참조하는 것이다.
이 경우에도 클로저를 사용하면 원하는 대로 동작하도록 만들 수 있다.
var i;
for (i = 0; i < 10; i++) {
(function(j) {
setTimeout(function() {
console.log(j);
}, 100);
})(i);
}
IIFE. **즉시 실행 함수 표현(IIFE, Immediately Invoked Function Expression)**은 정의되자마자 즉시 실행되는 Javascript Function
중간에 IIFE를 덧붙여 setTimeout()에 걸린 익명함수를 클로저로 만들었다.
앞서 말한대로 클로저는 만들어진 환경을 기억한다. 이 코드에서 i는 IIFE내에 j라는 형태로 주입되고, 클로저에 의해 각기 다른 환경속에 포함된다.
반복문이 10회 반복되므로 10개의 환경이 생길 것이고, 10개의 서로 다른 환경에 10개의 서로 다른 j가 생긴다.
이쯤에서 IIFE 매개변수로 i를 넘기지말고 그냥 직접 참좨도 되지 않느냐는 의문이 들 수 있다. 하지만 직접 그렇게 해보면 원하는대로 동작하지 않는다.
인자로 i를 넘기지 않는다면 당연히 클로저가 참조하는 IIFE의 함수 스코프에서도 i 값이 없으므로 생성 당시의 외부 스코프인 글로벌을 탐색하게 되고 결국 모두 같은 i를 참고하게 된다. 반면에 인자로 i를 넘기게되면 IIFE로 만든 10개의 스코프에 모두 i라는 변수가 다른 값으로 생기므로 정상적으로 동작할 수 있는 것이다.
참고로 여기서 콜백으로 넘기는 함수 자체를 IIFE로 만들면 되지 않냐는 사람도 있는데, 그렇게 하면 원하는대로 0-9까자ㅣ 출력은 되지만 함수 내부가 즉시 실행되어 버리므로 setTimeout()의 0.1초 딜레이가 작동하지 않는다.
클로저의 성능
클로저는 각자의 환경을 가진다. 이 환경을 기억하기 위해서는 당연히 메모리가 소모될 것이다.
클로저를 생성해놓고 참조를 제거하지 않는 것은 C++에서 동적할당으로 객체를 생성해놓고 delete를 사용하지 않는 것과 비슷하다. 클로저를 통해 내부 변수를 참조하는 동안에는 내부 변수가 차지하는 메모리를 GC가 회수하지 않는다.
따라서 클로저 사용이 끝나면 참조를 제거하는 것이 좋다.
function hello(name) {
var _name = name;
return function() {
console.log('Hello, ' + _name);
};
}
var hello1 = hello('승민');
var hello2 = hello('현섭');
var hello3 = hello('유근');
hello1(); // 'Hello, 승민'
hello2(); // 'Hello, 현섭'
hello3(); // 'Hello, 유근'
// 여기서 메모리를 release 시키기 클로저의 참조를 제거해야 한다.
hello1 = null;
hello2 = null;
hello3 = null;
이처럼 메모리 관리에 있어서 약점이 있지만 추가로 스코프 체인을 검색하는 시간과 새로운 스코프를 생성하는데 드는 비용도 감안하지 않을 수 없다.
출처: https://hyunseob.github.io/2016/08/30/javascript-closure/
JavaScript 클로저(Closure)
클로저란?MDN에서는 클로저를 다음과 같이 정의하고 있다. 클로저는 독립적인 (자유) 변수를 가리키는 함수이다. 또는, 클로저 안에 정의된 함수는 만들어진 환경을 ‘기억한다’. 흔히 함수 내
hyunseob.github.io
'개발 지식' 카테고리의 다른 글
this의 용법을 아는대로 설명하시오 (0) | 2022.11.11 |
---|---|
GET, POST 방식 차이 (0) | 2022.11.09 |
함수선언문과 함수표현식의 차이 (0) | 2022.11.03 |
호이스팅에 대해서 설명해 보세요. (0) | 2022.11.03 |
브라우저는 어떻게 동작하는가? (0) | 2022.11.03 |
댓글