본문 바로가기

프로그래밍/자바스크립트

자바스크립트 유효범위(javascript Scope), let, const



자바스크립트 유효범위(javascript Scope),  let, const 에 대해서 알아보겠습니다.


자바스크립트 유효범위 Scope란?

프로그래밍 언어에서 변수는 참조할 수 있는 범위가 존재합니다.

그래서 변수는 선언된 위치에 따라 유효범위를 갖게 됩니다.


C언어와 java같은 경우는 block level scope를 갖습니다.

block이란 { ... } 을 의미하며 이 블록을 벗어나면 블록 내부에 있는 변수를 참조할 수 없습니다.

private void foo(){

  int x = 10;

System.out.println(x);

위와 같은 코드는 에러를 발생합니다. foo 함수 내부에 존재하는 변수 x를 참조할 수 없기 때문이죠.


그러나 javascript의 경우에는 다릅니다.

자바스크립트에서 변수를 선언하는 방법에는 var, let, const 키워드가 있습니다.

지금부터 변수의 유효범위 개념과 변수 선언 방법에 따라 scope가 어떻게 바뀌는지 살펴보도록 하겠습니다.


함수 레벨 범위 ( Function Level Scope )

var 키워드를 사용하여 변수를 선언하면, 그 변수는 함수 레벨 범위를 갖습니다.

function foo(){

  if(true){

    var x = 10;

  }

  console.log(x);

}

 

foo();

console.log(x);

변수 x가 block level scope라면, if문이 끝날 때 변수 x에 대한 참조도 사라지게 됩니다.

그래서 foo() 함수 내부에 있는 console.log(x)는 출력 될 수 없을 것입니다.

그러나 var 키워드는 변수를 function level scope로 정의하기 때문에 if문이 끝나도 함수 내에서는 x가 유효하게 됩니다.

그런데 foo() 함수를 벗어나면 x에 대한 참조를 할 수 없기 때문에 foo() 함수 외부에 있는 console.log(x)는 에러가 발생합니다.


예제

if(true){

  var x = 10;

}

 

console.log(x);

이 코드는 10을 출력할까요? 에러를 발생할까요?

x는 if 내부에 선언되었지만, if는 함수가 아니기 때문에 x는 global scope 입니다.

따라서 정상적으로 10을 출력하게 됩니다.


블록 레벨 스코프 ( Block Level Scope )

let, const 키워드를 사용하여 변수를 선언하면, 그 변수는 블록 레벨 스코프를 갖습니다.

let으로 선언된 변수는 값을 수정할 수 있지만,

const로 선언된 변수는 값을 수정할 수 없는 상수입니다.


예제

// let 사용

if(true){

  let x = 10;

}

console.log(x);

 

// const 사용

if(true){

  const x = 10;

}

console.log(x);

위의 두 결과는 오류를 발생합니다.

let, const로 선언한 변수는 block level scope이기 때문에 if문을 벗어나면 x에 대해 참조할 수 없습니다.


const와 let의 특징

1) const는 상수라고 했습니다. 실제로 값을 바꿀 수 없는지 테스트 해보도록 하겠습니다.

예제

const x = 10;

x = 20;

console.log(20);

상수로 선언한 변수에 할당하려 했다는 오류가 발생합니다.


2) const는 선언하자마자 초기화해야 합니다.

예제

const x;

x = 10;

 

console.log(x)

const로 선언한 변수는 초기화 해야 한다는 오류가 발생합니다.


3) let은 var와 달리 중복 선언이 되지 않습니다.

예제

var x = 10;

var x = 20;

console.log(x);

var는 같은 변수에 대해 중복 선언이 가능하며, 그 결과는 20이 출력됩니다.


그러나 let은 어떨까요?

예제

let x = 10;

let x = 20;

console.log(x);

x는 이미 선언된 변수라는 오류를 발생합니다.


4) let을 사용하면 함수 내에서 반복문을 작성할 때 편리합니다.


예제

function count(){

  for(let i = 0; i < 10; i++){

    console.log(i);

  }

}

 

count();

let이 아닌 var 키워드를 작성했으면 클로저에 의해 10이라는 값이 9번 출력되었을 것입니다.


정적 스코프 ( Static Scope )또는 렉시컬 스코프 ( Lexical Scope )

정적 스코프는 실행되는 문맥에 의해 유효 범위가 결정되는 것을 의미합니다.

다음의 두 예제를 비교해보도록 하겠습니다.


예제1

var x = 10;

 

function foo(){

  var x = 20;

  goo();

}

 

function goo(){

  console.log(x);

}

 

foo();


예제2

var x = 10;

 

function foo(){

  var x = 20;

  goo();

 

  function goo(){

    console.log(x);

  }

}

 

foo();

예제1의 출력 결과는 10이지만, 예제2의 출력 결과는 20입니다.

이와 같이 함수가 선언된 시점에서 변수의 유효범위가 어떻게 결정되는지에 따라 값을 달리합니다.

이러한 유효 범위를 갖는 scope를 static scope 또는 lexical scope라 합니다.


이상으로 javascript에서의 변수 scope와 let, const 키워드에 대해 마치도록 하겠습니다.

var 키워드를 사용하는 function level scope와

let, const 키워드를 사용하는 block level scope가 있었습니다.