스코프(Scope)

NSH·2022년 5월 9일
0

JavaScript Concepts

목록 보기
6/8
post-thumbnail

스코프 란?

스코프는 변수, 함수의 이름과 같이 어떤 대상을 다른 대상과 구분하여 식별할 수 있는 유일한 이름을 찾아내기 위한 규칙이다. 자바스크립트는 이 규칙대로 식별자를 찾는다.

스코프의 구분

자바스크립트 스코프는 아래와 같이 구분된다.

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

모든 변수는 스코프를 가진다. 변수의 관점에서 스코프를 구분하면 아래와 같이 나눌 수 있다.

  • 전역 변수 ( Global variable )
    • 전역에 선언된 변수로 어디에서든지 참조할 수 있다.
  • 지역 변수 ( Local variable )
    • 함수(지역) 내에서 선언된 변수이다.
    • 함수 자신과 하위 함수에서만 참조할 수 있다.

변수는 선언 위치에 의해 스코프를 가지게 된다. 즉 전역에 선언된 변수는 전역 스코프를 갖는 전역 변수이고 함수(지역)에서 선언된 변수는 함수(지역)스코프를 갖는 변수가 된다.

자바스크립트 스코프 특징

자바스크립트는 함수 레벨 스코프를 따른다. 함수 레벨 스코프란 함수 코드 블록 내에서 선언된 변수는 함수 코드 블록 내에서만 유효하고 함수 외부에서는 유효하지 않다. 단 es6에 도입된 let, const를 사용하면 블록 레벨 스코프를 사용할 수 있다.

전역 스코프( Global scope )

전역에 변수를 선언하면 어디서든지 참조 가능한 전역 스코프를 가지는 전역 변수가 된다. var 키워드로 선언한 전역 변수는 전역 객체( window )의 프로퍼티이다.

var global = 'global';

function func() {
	var local = 'local';
	console.log(global); // global
	console.log(local);  // local
}

func();

console.log(global) // global
console.log(local)  // Uncaught ReferenceError: local is not defined 

전역 스코프에서는 함수(지역) 스코프에 선언된 local 변수를 참조할 수 없으므로 에러가 발생한다.

비 블록 레벨 스코프( Non block-level scope )

// Non block-level scope
if(true) {
	var x = 5;
}
console.log(x); // 5

변수 x는 코드 블록 내에서 선언되었다. 하지만 자바스크립트는 블록 레벨 스코프를 사용하지 않으므로 함수 밖에서 선언된 변수는 코드 블록 내에서 선언되면 모두 전역 스코프를 가진다. ( es6에 도입된 let, const를 사용하면 블록 레벨 스코프를 사용할 수 있다. )

블록 레벨 스코프( block-level scope )

// block-level scope es6 let, const
if(true) {
	let y = 5;
	console.log(y) // 5
}
console.log(y); // Uncaught ReferenceError: y is not defined

es6에 도입된 let, const는 블록 레벨 스코프를 가진다. 즉 변수 y는 if문 코드 블록 안에서 선언되었기 때문에 if문 코드 블록 안에서만 스코프를 가진다. 따라서 전역 스코프에서는 참조가 불가능하다.

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

자바스크립트는 함수 레벨 스코프를 사용한다. 함수 내에서 선언된 변수는 함수 외부에서는 참조할 수 없다.

var a = 'global'; // 전역 변수

(function func() {
	var b = 'local'; // 지역 변수
})();

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

함수 내에서 선언된 b는 지역 변수로 함수 외부에서 참조할 수 없기때문에 에러가 발생한다.

var x = 'global';

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

console.log(x); // global

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

렉시컬 스코프

var x = 1;

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

function bar() {
  console.log(x); // 지역 스코프에 참조할 x가 없으므로 전역 스코프의 x를 참조
}

foo(); // 1
bar(); // 1

위 예제의 실행 결과는 bar의 상위 스코프가 무엇인지에 따라서 결정된다. 함수를 어디서 호출하였는지에 따라 상위 스코프를 결정하는 동적 스코프와 함수를 어디 선언하였는지에 따라 상위 스코프를 결정하는 렉시컬 스코프 또는 정적 스코프가 존재하며 자바스크립트는 렉시컬 스코프를 따른다.

bar는 전역에 선언되었고 함수 bar의 상위 스코프는 전역 스코프이다. 따라서 x의 값인 1을 두번 출력한다.

암묵적 전역

var x = 10; // 전역 변수

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

foo(); // 30

선언하지 않은 식별자 y는 선언된 변수처럼 동작한다. 자바스크립트 엔진은 스코프 어디에서도 변수 y의 선언을 찾을 수 없으면 y = 20 을 window.y = 20 으로 해석하여 프로퍼티를 동적으로 생성한다. 따라서 y는 전역 객체의 프로퍼티가 되어 전역 변수처럼 동작한다.

하지만 y는 선언 없이 전역 객체의 프로퍼티로 추가된것으로 y는 변수가 아니다. 따라서 변수 호이스팅이 발생하지 않는다.

profile
잘 하고 싶다.

0개의 댓글