JS (24) - 클로저

최조니·2022년 7월 21일
0

JavaScript

목록 보기
21/36

클로저는 함수와 그 함수가 선언된 렉시컬 환경과의 조합


24.1 렉시컬 스코프

렉시컬 스코프(정적 스코프) :
자바스크립트 엔진은 함수를 어디서 호출했는지가 아니라 함수를 어디에 정의했는지에 따라 상위 스코프를 결정

  • 상위 스코프에 대한 참조는 함수 정의가 평가되는 시점에 함수가 정의된 환경(위치)에 의해 결정

24.2 함수 객체의 내부 슬롯 [[Environment]]

  • 함수가 정의된 위치와 호출된 위치는 다를 수 있음

  • 함수는 자신의 내부 슬롯 [[Environment]]에 자신이 정의된 환경 (상위 스코프의 참조)를 저장

  • 함수 객체의 내부 슬롯 [[Environment]]에 저장된 현재 실행 중인 실행 컨텍스트의 렉시컬 환경의 참조가 상위 스코프

  • 자신이 호출되었을 때 생성될 함수 렉시컬 환경의 외부 렉시컬 환경에 대한 참조에 저장될 참조값

  • 함수 객체는 내부 슬롯 [[Environment]]에 저장한 렉시컬 환경의 참조 (상위 스코프)를 자신이 존재하는 한 기억한다.


24.3 클로저와 렉시컬 환경

외부 함수보다 중첩 함수가 더 오래 유지되는 경우 중첩 함수는 이미 생명 주기가 종료한 외부 함수의 변수를 참조할 수 있음

→ 이러한 중첩 함수를 클로저


24.4 클로저의 활용

클로저는 상태를 안전하게 변경하고 유지하기 위해 사용
즉, 상태가 의도치 않게 변경되지 않도록 상태를 안전하게 은닉하고 특정 함수에게만 상태 변경을 허용하기 위해 사용

const increase = (function () {
  let num = 0;
  
  return function () {
    return ++num;
  };
}());

console.log(increase());	// 1
console.log(increase());	// 2
console.log(increase());	// 3
const counter = (function () {
  // 카운트 상태 변수
  let num = 0;
  
  // 클로저인 메서드를 갖는 객체 반환
  return {
    increase() {
      return ++num;
    },
    decrease() {
      return num > 0 ? --num : 0;
    }
  };
}());

console.log(increase());	// 1
console.log(increase());	// 2
    
console.log(decrease());	// 1
console.log(decrease());	// 0

24.5 캡슐화와 정보 은닉

캡슐화
: 객체의 상태를 나타내는 프로퍼티와 프로퍼티를 참조하고 조작할 수 있는 동작인 메서드를 하나로 묶는 것

  • 캡슐화는 객체의 특정 프로퍼티나 메서드를 감출 목적으로 사용하기도 함 (정보 은닉)

정보 은닉
외부에 공개할 필요가 없는 구현의 일부를 외부에 공개되지 않도록 감추어 적절치 못한 접근으로부터 객체의 상태가 변경되는 것을 방지해 정보를 보호하고, 객체 간의 상호 의존성 (결합도)를 낮추는 효과가 있음


24.6 자주 발생하는 실수

var funcs = [];

for (var i = 0; i < 3; i++) {
  funcs[i] = function () { return i; };
}

for (var j = 0; j < funcs.length; j++) {
  console.log(funcs[j]());
}
var funcs = [];

for (var i = 0; i < 3; i++) {
  funcs[i] = (function (id) {
    return function() {
      return id;
    };
  }(i));
}

for (var j = 0; j < funcs.length; j++) {
  console.log(funcs[j]());
}
profile
Hello zoni-World ! (◍ᐡ₃ᐡ◍)

0개의 댓글