2. 프로그래밍 언어 공부/JavaScript

[공부 필기] JavaScript 문법 공부 16일차 (1)

김간장 2022. 5. 7. 15:47

※ 필자는 초초초보자입니다.

※ 틀린 내용에 대한 피드백은 언제든지 환영합니다.

 

 

공부하고 있는 자료 : modern JavaScript tutorial 

https://ko.javascript.info/function-expressions

 

함수 표현식

 

ko.javascript.info

 

✅ 함수 표현식

- 자바스크립트는 함수를 특별한 종류의 값으로 취급한다.

   » 다른 언어에서는 '특별한 동작을 하는 구조'로 취급하지만, 자바스크립트에서는 아니다.

 

- 이전에서 배운 함수 선언문 방식 (예시)

function pow(base, factor) {
	let result = 1;

	for (let i = 0; i < factor; i++) {
		result *= base;
	}
	return (result);
}

 

- 함수 선언 방식 외에 함수 표현식을 사용해서 함수를 만들 수 있다.

- 함수 표현식 예시

let pow = function(base, factor) {
	let result = 1;

	for (let i = 0; i < factor; i++) {
		result *= base;
	}
	return (result);
};

- 위의 코드는 함수가 변수에 할당된 것

 

- 함수가 어떤 방식으로 만들어졌는지 관계없이 함수는 값(특별한 종류의 값)이고, 변수에 할당할 수 있다.

- 참고로 자바스크립트는 "괄호"가 있어야만 함수가 호출된다. 매개변수가 없어도(=void) 괄호는 있어야 한다.

 

# 만약 괄호 없는 함수를 화면에 출력해보면 다음과 같은 모습이 나온다.

let pow = function(base, factor) {
	let result = 1;

	for (let i = 0; i < factor; i++) {
		result *= base;
	}
	return (result);
};

console.log(pow);

 

 

- 함수는 결국 값이기 때문에, 값에 할 수 있는 일을 함수에도 할 수 있다.

- 예를 들면, 함수 복사!

let pow = function(base, factor) {
	let result = 1;

	for (let i = 0; i < factor; i++) {
		result *= base;
	}
	return (result);
};

let copyPow = pow; // 함수 복사

console.log(copyPow(3, 2)); // 9

console.log(pow(3, 2)); // 9

   » 위의 코드 과정

   » 처음에 pow라는 함수를 생성하고 pow라는 변수에 저장된다.

   » pow를 새로운 변수 copyPow에 복사한다. (이때, pow 뒤에 소괄호가 없다는 것을 유의해야한다)

      만약 소괄호를 붙여버리면 함수 그 자체가 아니라, 함수 호출 결과가 copyPow에 저장되어버린다.

 

 

# 함수 표현식에는 끝에 세미 콜론이 붙는다.

let pow = function(base, factor) {
	let result = 1;

	for (let i = 0; i < factor; i++) {
		result *= base;
	}
	return (result);
}; // 세미콜론 붙임

- 왜 붙이냐면 함수 표현식은 코드 블록이 아니고, 값처럼 취급되어 변수에 할당되는 것이기 때문이다.

- if { ... } , for { ... } 등은 코드 블록이며 끝에 세미콜론이 없어도 된다.

 

 

✅ 콜백 함수

- 매개변수가 3개 있는 함수 ask가 있다.

   » 여기에서 ask의 매개변수 3개 중 두번째, 세번째는 함수(function)라고 가정한다.

function ask(question, yes, no) {
	if (confirm(question)) yes();
	else no();
}
...

ask("동의하십니까?", showOk, showCancel);

// 출처 : 모던 자바스크립트 https://ko.javascript.info/arrow-functions-basics

 

- ask 함수의 두번째 인자인 showOk와 세번째 인자인 showCancel를 콜백 함수(또는 콜백)이라고 부른다.

 

- 콜백함수는

   » 함수를 함수의 인자로 전달

   » 필요하다면 인자로 전달한 함수를 '나중에 호출(called back)' 하는 것이다.

- 위의 코드는 사용자가 "yes"를 대답하면, showOk라는 함수가 콜백되고,

   "no"라고 대답하면, showCancel이라는 함수가 콜백된다.

 

- 아래와 같이 짧게 작성할 수도 있다.

  » 이때, 두번째와 세번째 인자(함수)는 익명함수라고 한다.

 

function ask(question, yes, no) {
	if (confirm(question)) yes();
	else no();
}

ask("동의하십니까?", function() { alert("동의하셨습니다."); }, function() { alert("취소 버튼을 누르셨습니다."); });

// 출처 : 모던 자바스크립트 (https://ko.javascript.info/function-expressions)

 

- (익명함수이자) ask의 두번째, 세번째 인자는 ask 바깥에선 접근할 수 없다.

- 위의 코드는 그렇게 의도를 가지고 구현했기 때문에 바깥에서 접근할 수 없어도 문제가 없는 코드라고 한다.

 

 

※ 함수는 동작을 나타내는

- 문자열, 숫자가 일반적인 값("Hello", 1, 0, true 등)을 나타내는 데이터라면

- 함수는 하나의 "동작"을 나타내는 값이다.

 

 함수 표현식 vs 함수 선언문

- 두 표현 방법의 차이는 아래와 같다.

 

- 첫번째 : 문법

   » 함수 선언문 : 주요 코드 흐름 중간에 독자적인 구문 형태로 존재한다.

   » 함수 표현식 : 표현식이나 구문 구성(syntax construct) 내부에 생성된다.

 

- 두번째 : 자바스크립트 엔진이 언제 함수를 생성하는가

  » 함수 선언문 : 함수 선언문이 정의되기 전에 호출될 수도 있다.

      (즉, 함수 표현식과 다르게 함수 선언문이 스크립트 어디에 있어도 상관없이 사용할 수 있다.)

  » 함수 표현식 : 실제 실행 흐름이 해당 함수에 도달했을 때 함수를 생성한다.

      (let pow = function { ... }가 있다면, 우측에 도달했을 때, 함수가 생성된다.

       그 이후부터 해당 함수를 사용할 수 있다.)

 

- 세번째 : 스코프

   » 엄격 모드(use strict)에서 함수 선언문 :

       코드 블록 내에 위치하면, 해당 함수는 블록 내 어디서든 접근할 수 있다.

       하지만 블록 밖에서는 함수에 접근하지 못한다.

   » 아래의 코드는 if { ... } else { ... } 코드 블록 안에서 함수 선언문을 통해 조건에 따라 다르게 함수를 정의했으며

       해당 코드 블록 안에서만 유효하다.

   » 만약 아래의 코드를 실행하면, prompt가 실행되며 if의 코드 혹은 else의 코드 중 하나가 실행된다.

       하지만, isLimit();는 동작하지 않는다.

"use strict";

if (prompt("나이를 입력하세요", "1") < 20) {
	function isLimit() {
		alert("최대 한도가 제한됩니다.");
	}
}
else {
	function isLimit() {
		alert("제한 없이 사용할 수 있습니다.");
	}
}

console.log(isLimit); // isLimit is not defined (Error)

 

   » 엄격 모드(use strict)에서의 함수 표현식 :

       변수에 함수를 할당하기 때문에, 전역 변수(외부 변수)에 함수를 할당하게 되면 코드 블록 밖에서도 접근할 수 있다.

"use strict";

let isLimit;

if (prompt("나이를 입력하세요", "1") < 20) {
	isLimit = function() {
		alert("최대 한도가 제한됩니다.");
	}
}
else {
	isLimit = function() {
		alert("제한 없이 사용할 수 있습니다.");
	}
}

console.log(isLimit);

 

 

※ 두번째와 같은 현상이 발생하는 이유는 자바스크립트의 내부 알고리즘 때문이라고 한다.

자바스크립트는 스크립트를 실행하기 전,

- 준비단계에서 전역에 선언된 함수 선언문을 찾고

- 해당 함수를 생성한다.

(스크립트가 진짜 실행되기 전, 초기화 단계를 수행하는데, 이때 함수 선언 방식으로 정의한 함수를 생성한다.)

- 그 다음에 스크립트가 진짜로 실행된다.