티스토리 뷰

JAVASCRIPT

[javascript] 스코프

jeongah story 2020. 4. 12. 19:16

스코프

: 참조 대상 식별자(변수, 함수의 이름과 같이 어떤 대상을 다른 대상과 구분하여 식별할  있는 유일한 이름)  찾아내기 위한 규칙! 

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)
전역에서 선언된 변수이며 어디에든 참조할 수 있다.
지역 변수 (Local variable)
지역(함수) 내에서 선언된 변수이며 그 지역과 그 지역의 하부 지역에서만 참조할 수 있다.

 

변수는 선언 위치(전역 또는 지역)에 의해 스코프를 가지게 됩니다.

, 전역에서 선언된 변수는 전역 스코프를 갖는 전역 변수이고, 지역(자바스크립트의 경우 함수 내부)에서 선언된 변수는 지역 스코프를 갖는 지역 변수가 됩니다.

 

 

* 전역스코프 

: 코드 어디에서든지 참조할 수 있다.

var global = 'global';

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

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

변수 global는 함수 영역 밖의 전역에서 선언되었습니다.

 

이처럼, 전역 변수의 사용은 변수 이름이 중복 있고

의도치 않은 재할당에 의한 상태 변화로 코드를 예측하기 어렵게 만드므로 사용을 억제하여야 합니다.

 

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

 

(function () {
  var MYAPP = {};

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

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

console.log(MYAPP.student.name);

전역변수 사용을 억제하기 위해,

즉시 실행 함수(IIFE, Immediately-Invoked Function Expression)를 사용할 수 있다.

이 방법을 사용하면 전역변수를 만들지 않으므로 라이브러리 등에 자주 사용된다.

즉시 실행 함수는 즉시 실행되고 그 후 전역에서 바로 사라진다.

 

 

스코프의 특징

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


let y = 0;
{
  let y = 1;
  console.log(y); // 1
}
console.log(y);   // 0

const z = 0;
{
  const z = 1;
  console.log(z); // 1
}
console.log(z);   // 0

 

var

코드 블록 내의 변수 x는 전역 변수이기 때문에 전역에서 선언된 전역 변수 x의 값 0을 새로운 값 1으로 재할당하여 덮어쓴다.

 

let

코드 블록 내에 선언된 변수 y는 블록 레벨 스코프를 갖는 지역 변수이다.

전역에서 선언된 변수 y와는 다른 별개의 변수이다. 따라서 전역에서는 변수 y를 참조할 수 없다

 

 

자바스크립트는 함수 레벨 스코프 따릅니다. 

함수 레벨 스코프란 함수 코드 블록 내에서 선언된 변수는 함수 코드 블록 내에서만 유효하고 함수 외부에서는

유효하지 않는(참수 할 수 없다.) 것입니다.

 

단, ECMAScript6에 도입된 let, const keyword 를 사용하면 블록 레벨 스코프를 사용할수 있습니다.

블록 레벨 스크포란, 코드블록 ({…}) 내에서 유효한 스코프를 의미합니다. 여기서 유효하다 라는건참조(접근))” 수있다는 뜻입니다.

 

 

 

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

 

if (true) {
  var x = 5;  //전역변수
}
console.log(x);  //5

 

 

변수 x는 코드 블록 내에서 선언되었습니다.

하지만 자바스크립트는 블록 레벨 스코프를 사용하지 않으므로 

함수 밖에서 선언된 변수는 코드 블록 내에서 선언되었다할지라도 모두 전역 스코프을 갖게된다.

따라서 변수 x는 전역 변수이다.

 

* 함수 레벨 스코프(functon-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); //local

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

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

 

내부함수는 자신을 포함하고 있는 외부함수의 변수에 접근할 수 있다. 이는 매우 유용하다.

함수 bar에서 참조하는 변수 x는 함수 foo에서 선언된 지역변수이다.

이는 실행 컨텍스트의 스코프 체인에 의해 참조 순위에서 전역변수 x가 뒤로 밀렸기 때문이다.

 

 

var x = 10;

function foo() {
  x = 100;
  console.log(x); 
}
foo(); ////100
console.log(x); // 100

함수(지역) 영역에서 전역변수를 참조할 수 있으므로 전역변수의 값도 변경할 수 있다.

내부 함수의 경우, 전역변수는 물론 상위 함수에서 선언한 변수에 접근/변경이 가능하다.

 

var x = 10;

function foo(){
  var x = 100;
  console.log(x);  //100

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

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

중첩 스코프는 가장 인접한 지역을 우선하여 참조한다.

 

 

 

 

 

 

렉시컬 스코프 (어휘적 범위 지정(Lexical scoping))

클로저를 이해하려면 자바스크립트가 어떻게 변수의 유효범위를 지정하는지(Lexical scoping)를 먼저 이해해야 한다.

 

function init() {
  var name = "Mozilla"; // name은 init에 의해 생성된 지역 변수이다.
  function displayName() { // displayName() 은 내부 함수이며, 클로저다.
    alert(name); // 부모 함수에서 선언된 변수를 사용한다.
  }
  displayName();
}
init();

displayName() 내부엔 자신만의 지역 변수가 없습니다.
중첩된 함수는 외부 범위(scope)에서 선언한 변수에도 접근할 수 있습니다.

 

var x = 1;

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

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

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

렉시컬 스코프는 함수를 어디서 호출하는지가 아니라 어디에 선언하였는지에 따라 결정됩니다.

위 예제의 함수 bar는 전역에 선언되었습니다.

따라서 함수 bar의 상위 스코프는 전역 스코프이고 위 예제는 전역 변수 x의 값 1을 두번 출력합니다.

 

 

 

암묵적 전역

 

var x = 10; // 전역 변수

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

foo(); // 30

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

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

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

위 예제의 y는 선언하지 않은 식별자이다.

따라서 y = 20이 실행되면 참조 에러가 발생할 것처럼 보인다.

하지만 선언하지 않은 식별자 y는 마치 선언된 변수처럼 동작한다.

이는 선언하지 않은 식별자에 값을 할당하면 전역 객체의 프로퍼티가 되기 때문이다.

 

자바스크립트 엔진은 y = 20을 window.y = 20으로 해석하여 프로퍼티를 동적 생성한다.

결국 y는 전역 객체의 프로퍼티가 되어 마치 전역 변수처럼 동작한다.

이러한 현상을 암묵적 전역(implicit global)이라 한다.

 

하지만 y는 변수 선언없이 단지 전역 객체의 프로퍼티로 추가되었을 뿐이다.

따라서 y는 변수가 아니다. 따라서 변수가 아닌 y는 변수 호이스팅이 발생하지 않는다.

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

 

 

 

 

 

 

참고 사이트 : https://poiemaweb.com/js-scope 

참고 사이트 : https://developer.mozilla.org/ko/docs/Web/JavaScript/Guide/Closures

 

클로저

클로저는 함수와 함수가 선언된 어휘적 환경의 조합이다. 클로저를 이해하려면 자바스크립트가 어떻게 변수의 유효범위를 지정하는지(Lexical scoping)를 먼저 이해해야 한다.

developer.mozilla.org

 

Scope | PoiemaWeb

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

poiemaweb.com

 

'JAVASCRIPT' 카테고리의 다른 글

[Javascript] Set  (0) 2020.04.19
[Javascript] Map  (0) 2020.04.19
[javascript] ECMAScript6의 배열과 새롭게 추가된 기능  (0) 2020.04.11
[javascript30] 시계  (0) 2020.04.08
[javascript30] 드럼치기  (0) 2020.04.07
댓글
© 2018 eh2world