[JavaScript] 스코프

Letmegooutside·2022년 1월 18일
0

JavaScript

목록 보기
9/25
post-thumbnail
post-custom-banner

스코프

변수를 찾아내기 위한 규칙, 변수에 접근할 수 있는 범위

변수는 전역 또는 코드 블록(if, for, while, try/catch 등)이나 함수 내에 선언하며 코드 블록이나 함수는 중첩될 수 있다.
식별자는 자신이 어디에서 선언됐는지에 의해 다른 코드가 자신을 참조할 수 있는 범위를 갖는다.

var x = 'global';

function foo () {
  var x = 'function scope';
  console.log(x);
}

foo(); // function scope
console.log(x); // global

이 예제에서 전역에 선언된 변수 x는 어디에든 참조할 수 있다.
하지만 함수 foo 내에서 선언된 변수 x는 함수 foo 내부에서만 참조할 수 있고 함수 외부에서는 참조할 수 없다. 이러한 규칙을 스코프라고 한다.
만약 스코프가 없다면 같은 식별자 이름은 충돌을 일으키므로 프로그램 전체에서 하나밖에 사용할 수 없게 된다.

스코프의 구분

자바스크립트에서 스코프는 2가지로 나눌 수 있다.

  • 전역 스코프 (Global scope)
    코드 어디에서든지 참조할 수 있다.
  • 지역 스코프 (Local scope or Function-level scope)
    함수 코드 블록이 만든 스코프로 함수 자신과 하위 함수에서만 참조할 수 있다.

모든 변수는 스코프를 갖는다. 변수의 관점에서 스코프를 구분하면 다음과 같이 2가지로 나눌 수 있다.

  • 전역 변수 (Global variable)
    전역에서 선언된 변수이며 어디에든 참조할 수 있다.
    (자바스크립트에서 var 키워드로 선언한 전역 변수는 전역 객체(Global Object) window의 프로퍼티이다.)

  • 지역 변수 (Local variable)
    지역(함수) 내에서 선언된 변수이며 그 지역과 그 지역의 하부 지역에서만 참조할 수 있다.

변수는 선언 위치(전역 또는 지역)에 의해 스코프를 가지게 된다.
즉 전역 스코프를 갖는 전역 변수는 전역에서 참조할 수 있고, 지역에서 선언된 지역 변수는 그 지역과 그 지역의 하부 지역에서만 참조할 수 있다.

자바스크립트 스코프의 특징

  • 블록 레벨 스코프: 코드 블록({…})내에서 유효한 스코프를 의미

  • 함수 레벨 스코프 : 함수 코드 블록 내에서 선언된 변수는 함수 코드 블록 내에서만 유효하고 함수 외부에서는 참조할 수 없다는 것

대부분의 C-family language는 블록 레벨 스코프(block-level scope)를 따른다.

int main(void) {
  // block-level scope
  if (1) {
    int x = 5;
    printf("x = %d\n", x);
  }

  printf("x = %d\n", x); // use of undeclared identifier 'x'

  return 0;
}

따라서 위의 C언어 코드를 보면 if문 내에서 선언된 변수 xif문 코드 블록 내에서만 유효하다. 즉, if문 코드 블록 밖에서는 참조가 불가능하다.

하지만 자바스크립트는 함수 레벨 스코프(function-level scope)를 따르기 때문에 다른 코드 블록에 상관없이 함수 코드 블록 내에서만 유효하다.
(단, ECMAScript 6에서 도입된 let 키워드를 사용하면 블록 레벨 스코프를 사용할 수 있다.)

var x = 0;
{
  var x = 1;
  console.log(x); // 1
}
console.log(x);   // 1

// let 키워드 사용 시
let y = 0;
{
  let y = 1;
  console.log(y); // 1
}
console.log(y);   // 0

전역 스코프(Global scope)

C언어의 경우 main 함수가 시작점이 되기 때문에 대부분은 코드는 main 함수 내에 포함된다.
따라서 전역 변수를 선언하기 위해서는 의도적으로 main 함수 밖에 변수를 선언하여야 한다.

#include <stdio.h>

/* global variable declaration */
int g;

int main () {

  // local variable declaration
  int a, b;

  // actual initialization
  a = 10;
  b = 20;
  g = a + b;

  printf ("value of a = %d, b = %d and g = %d\n", a, b, g);

  return 0;
}

하지만 자바스크립트는 다른 C-family language와는 달리 특별한 시작점이 없으며 코드가 나타나는 즉시 해석되고 실행된다.
따라서 전역에 변수를 선언하기 쉬우며 이것는 전역 변수를 남발하게 하는 문제를 야기시킨다.

전역 변수의 사용은 변수 이름이 중복될 수 있고, 의도치 않은 재할당에 의한 상태 변화로 코드를 예측하기 어렵게 만드므로 사용을 억제하여야 한다.

함수 레벨 스코프(Function-level scope)

var a = 10;     // 전역변수

(function () {
  var b = 20;   // 지역변수
})();

console.log(a); // 10
console.log(b); // "b" is not defined

자바스크립트는 함수 레벨 스코프를 사용한다.
즉, 함수 내에서 선언된 매개변수와 변수는 함수 외부에서는 유효하지 않으므로 변수 b는 지역 변수이다.

var x = 'global';

function foo() {
  var x = 'local';
  console.log(x);
}

foo();          // local
console.log(x); // global

전역변수 x와 지역변수 x가 중복 선언되었다.
전역 영역에서는 전역변수만이 참조 가능하고 함수 내 지역 영역에서는 전역과 지역 변수 모두 참조 가능하나 위 예제와 같이 변수명이 중복된 경우, 지역변수를 우선하여 참조한다.

내부 함수의 경우

var x = 'global';

function foo() {
  var x = 'local';
  console.log(x);

  function bar() {  // 내부함수
    console.log(x); // ?
  }

  bar();
}
foo();
console.log(x); // ?

내부함수는 자신을 포함하고 있는 외부함수의 변수에 접근할 수 있다.
따라서 클로저에서와 같이 내부함수가 더 오래 생존하는 경우, 다른 언어와는 다른 움직임을 보인다.

함수 bar에서 참조하는 변수 x는 함수 foo에서 선언된 지역변수이다.
이는 실행 컨텍스트의 스코프 체인에 의해 참조 순위에서 전역변수 x가 뒤로 밀렸기 때문이다.
즉 중첩 스코프일 경우는 가장 인접한 지역을 우선하여 참조한다.

렉시컬 스코프

var x = 1;

function foo() {
  var x = 10;
  bar();
}

function bar() {
  console.log(x);
}

foo(); // ?
bar(); // ?

위 예제의 실행 결과는 함수 bar의 상위 스코프가 무엇인지에 따라 결정되며 아래의 두 가지 패턴으로 예측할 수 있다.
프로그래밍 언어는 이 두가지 방식 중 하나의 방식으로 함수의 상위 스코프를 결정한다.

  • 함수를 어디서 호출하였는지에 따라 상위 스코프를 결정
    함수 bar의 상위 스코프는 함수 foo와 전역일 것이다.
    이 방식을 동적 스코프(Dynamic scope)라고 한다.

  • 함수를 어디서 선언하였는지에 따라 상위 스코프를 결정
    함수 bar의 스코프는 전역일 것이다.
    이 방식을 렉시컬 스코프(Lexical scope) 또는 정적 스코프(Static scope)라 한다.
    자바스크립트를 비롯한 대부분의 프로그래밍 언어는 렉시컬 스코프를 따른다.

렉시컬 스코프

렉시컬 스코프는 함수를 어디서 호출하는지가 아니라 어디에 선언하였는지에 따라 결정된다.
자바스크립트는 렉시컬 스코프를 따르므로 함수를 선언한 시점에 상위 스코프가 결정된다.
위 예제의 함수 bar는 전역에 선언되었으므로 함수 bar의 상위 스코프는 전역 스코프이고 위 예제는 전역 변수 x의 값 1을 두번 출력한다.

암묵적 전역

var x = 10; // 전역 변수

function foo () {
  // 선언하지 않은 식별자
  y = 20;
  console.log(x + y);
}

foo(); // 30

위 예제에서 선언하지 않은 식별자 y는 마치 선언된 변수처럼 동작한다.
이는 선언하지 않은 식별자에 값을 할당하면 전역 객체의 프로퍼티가 되기 때문이다.

foo 함수가 호출되면 자바스크립트 엔진은 변수 y에 값을 할당하기 위해 먼저 스코프 체인을 통해 선언된 변수인지 확인한다.
이 때 foo 함수의 스코프와 전역 스코프 어디에서도 변수 y의 선언을 찾을 수 없으므로 참조 에러가 발생해야 할 것 같다.
하지만 자바스크립트 엔진은 y = 20window.y = 20으로 해석하여 프로퍼티를 동적 생성한다.
결국 y는 전역 객체의 프로퍼티가 되어 마치 전역 변수처럼 동작하는데 이러한 현상을 암묵적 전역(implicit global)이라 한다.

y는 변수 선언없이 단지 전역 객체의 프로퍼티로 추가되었을 뿐이다.
따라서 y는 변수가 아니므로 변수 호이스팅이 발생하지 않는다.

또한 변수가 아니라 단지 프로퍼티인 ydelete 연산자로 삭제할 수 있다.
전역 변수는 프로퍼티이지만 delete 연산자로 삭제할 수 없다.

var x = 10; // 전역 변수

function foo () {
  // 선언하지 않은 변수
  y = 20;
  console.log(x + y);
}

delete x; // 전역 변수는 삭제되지 않는다.
delete y; // 프로퍼티는 삭제된다.

console.log(window.x); // 10
console.log(window.y); // undefined

최소한의 전역변수 사용

전역변수 객체 사용

전역변수 사용을 최소화하는 방법 중 하나는 애플리케이션에서 전역변수 사용을 위해 다음과 같이 전역변수 객체 하나를 만들어서 사용하는 것이다. (더글라스 크락포드의 제안)

var MYAPP = {};

MYAPP.student = {
  name: 'Lee',
  gender: 'male'
};

console.log(MYAPP.student.name);

즉시실행함수를 이용한 전역변수 사용 억제

전역변수 사용을 억제하기 위해, 즉시 실행 함수(IIFE, Immediately-Invoked Function Expression)를 사용할 수 있다.
이 방법을 사용하면 전역변수를 만들지 않으므로 라이브러리 등에 자주 사용된다.
즉시 실행 함수는 즉시 실행되고 그 후 전역에서 바로 사라진다.

(function () {
  var MYAPP = {};

  MYAPP.student = {
    name: 'Lee',
    gender: 'male'
  };

  console.log(MYAPP.student.name); // Lee
}());

console.log(MYAPP.student.name); // Reference Error



Reference
https://poiemaweb.com/js-scope#9 이 내용을 정리

post-custom-banner

0개의 댓글