함수
스크립트를 작성하다 보면 유사한 동작을 하는 코드가 여러 곳에서 필요할 때가 많습니다.
사용자가 로그인이나 로그아웃을 했을 때 안내 메시지를 보여주는 동작 같은 경우 말이죠.
함수는 프로그램을 구성하는 주요 '구성 요소(building block)'입니다. 함수를 이용하면 중복 없이 유사한 동작을 하는 코드를 여러 번 호출할 수 있습니다.
우리는 앞서 다양한 예시에서 alert(message)
, prompt(message, default)
, confirm(question)
과 같은 내장 함수를 사용해 보았습니다. 이번 챕터에선 함수를 직접 만드는 방법에 대해 알아보겠습니다.
함수 선언(function declaration) 방식을 이용하면 함수를 만들 수 있습니다(함수 선언 방식은 함수 선언문이라고 부르기도 합니다 – 옮긴이).
함수 선언 방식은 아래와 같이 작성할 수 있습니다.
function
키워드, 함수 이름, 괄호로 둘러싼 매개변수를 차례로 써주면 함수를 선언할 수 있습니다. 위 함수에는 매개변수가 없는데, 만약 매개변수가 여러 개 있다면 각 매개변수를 콤마로 구분해 줍니다. 이어서 함수를 구성하는 코드의 모임인 '함수 본문(body)'을 중괄호로 감싸 붙여줍시다.
새롭게 정의한 함수는 함수 이름 옆에 괄호를 붙여 호출할 수 있습니다. showMessage()
같이 말이죠.
예시:
showMessage()
로 함수를 호출하면 함수 본문이 실행됩니다. 위 예시에선 showMessage를 두 번 호출했으므로 얼럿 창이 두 번 뜹니다.
함수의 주요 용도 중 하나는 중복 코드 피하기입니다. 위 예시를 통해 이를 확인해 보았습니다.
얼럿 창에 보여줄 메시지를 바꾸거나 메시지를 보여주는 방식 자체를 변경하고 싶다면, 함수 본문 중 출력에 관여하는 코드 딱 하나만 수정해주면 됩니다.
함수 내에서 선언한 변수인 지역 변수(local variable)는 함수 안에서만 접근할 수 있습니다.
예시:
함수 내부에서 함수 외부의 변수인 외부 변수(outer variable)에 접근할 수 있습니다.
함수에선 외부 변수에 접근하는 것뿐만 아니라, 수정도 할 수 있습니다.
예시:
외부 변수는 지역 변수가 없는 경우에만 사용할 수 있습니다.
함수 내부에 외부 변수와 동일한 이름을 가진 변수가 선언되었다면, 내부 변수는 외부 변수를 가립니다. 예시를 살펴봅시다. 함수 내부에 외부 변수와 동일한 이름을 가진 지역 변수 userName
가 선언되어 있습니다. 외부 변수는 내부 변수에 가려져 값이 수정되지 않았습니다.
전역 변수
위 예시의 userName
처럼, 함수 외부에 선언된 변수는 전역 변수(global variable) 라고 부릅니다.
전역 변수는 같은 이름을 가진 지역 변수에 의해 가려지지만 않는다면 모든 함수에서 접근할 수 있습니다.
변수는 연관되는 함수 내에 선언하고, 전역 변수는 되도록 사용하지 않는 것이 좋습니다. 비교적 근래에 작성된 코드들은 대부분 전역변수를 사용하지 않거나 최소한으로만 사용합니다. 다만 프로젝트 전반에서 사용되는 데이터는 전역 변수에 저장하는 것이 유용한 경우도 있으니 이 점을 알아두시기 바랍니다.
매개변수(parameter)를 이용하면 임의의 데이터를 함수 안에 전달할 수 있습니다. 매개변수는 인자(parameter) 라고 불리기도 합니다.
아래 예시에서 함수 showMessage는 매개변수 from
과 text
를 가집니다.
(*)
, (**)
로 표시한 줄에서 함수를 호출하면, 함수에 전달된 인자는 지역변수 from
과 text
에 복사됩니다. 그 후 함수는 지역변수에 복사된 값을 사용합니다.
예시 하나를 더 살펴봅시다. 전역 변수 from
이 있고, 이 변수를 함수에 전달하였습니다. 함수가 from
을 변경하지만, 변경 사항은 외부 변수 from
에 반영되지 않았습니다. 함수는 언제나 복사된 값을 사용하기 때문입니다.
함수의 매개변수에 전달된 값을 *인수(argument)*라고 부르기도 합니다.
더 정확한 이해를 돕기 위해 용어를 다시 한번 정리해볼까요?
매개변수는 함수 선언 방식 괄호 사이에 있는 변수입니다(선언 시 쓰이는 용어).
인수는 함수를 호출할 때 매개변수에 전달되는 값입니다(호출 시 쓰이는 용어).
즉, 함수 선언 시 매개변수를 나열하게 되고, 함수를 호출할 땐 인수를 전달해 호출합니다.
위 예에서 함수 showMessage
는 from
과 text
라는 두 매개변수를 사용해 선언되었고, 그 후 호출 시엔 from
, Hello
라는 두 인수를 사용해 호출되었습니다.
함수 호출 시 매개변수에 인수를 전달하지 않으면 그 값은 undefined
가 됩니다.
예시를 통해 이에 대해 알아봅시다. 위에서 정의한 함수 showMessage(from, text)
는 매개변수가 2개지만, 아래와 같이 인수를 하나만 넣어서 호출할 수 있습니다.
이렇게 코드를 작성해도 에러가 발생하지 않습니다. 두 번째 매개변수에 값을 전달하지 않았기 때문에 text
엔 undefined
가 할당될 뿐입니다. 따라서 에러 없이 "Ann: undefined"
가 출력됩니다.
매개변수에 값을 전달하지 않아도 그 값이 undefined
가 되지 않게 하려면 함수를 선언할 때 =
를 사용해 '기본값(default value)'을 설정해주면 됩니다.
이젠 text
가 값을 전달받지 못해도 undefined
대신 기본값 "no text given"
이 할당됩니다.
매개변수에 값을 전달해도 그 값이 undefined
와 엄격히 일치한다면 기본값이 할당됩니다.
예시:
위 예시에선 문자열 "no text given"
을 기본값으로 설정했습니다. 하지만 아래와 같이 복잡한 표현식도 기본값으로 설정할 수도 있습니다.
매개변수 기본값 평가 시점
자바스크립트에선 함수를 호출할 때마다 매개변수 기본값을 평가합니다. 물론 해당하는 매개변수가 없을 때만 기본값을 평가하죠.
위 예시에선 매개변수 text
에 값이 전달되는 경우 anotherFunction()
은 호출되지 않습니다.
반면 text
에 값이 없는 경우 showMessage()
를 호출할 때마다 anotherFunction()
이 호출됩니다.
구식 자바스크립트에서 매개변수 기본값 설정하는 방법
몇 년 전만 해도 자바스크립트엔 매개변수 기본값 관련 구문이 없었습니다. 그래서 매개변수 기본값을 설정하려면 다른 방법을 사용해야만 했죠.
요즘에도 오래된 스크립트를 보다 보면 매개변수 기본값 설정 관련 코드를 접할 수 있습니다.
구식 코드에서는 매개변수 기본값 설정을 위해 먼저 매개변수 값이 undefined
인지 명시적으로 확인하고, 일치하는 경우엔 기본값을 설정합니다.
예시:
이 방법 말고도 논리 연산자 ||
를 사용해 매개변수 기본값을 설정하는 방법도 있습니다.
가끔은 함수를 선언할 때가 아닌 함수 선언 후에 매개변수 기본값을 설정하는 것이 적절한 경우도 있습니다.
이런 경우엔 함수를 호출할 때 매개변수를 undefined
와 비교하여 매개변수가 전달되었는지를 확인합니다.
이렇게 if
문을 쓰는 것 대신 논리 연산자 ||
를 사용할 수도 있습니다.
이 외에도 모던 자바스크립트 엔진이 지원하는 nullish 병합 연산자(nullish coalescing operator) ??
를 사용하면 0
처럼 falsy로 평가되는 값들을 일반 값처럼 처리할 수 있어서 좋습니다.
함수를 호출했을 때 함수를 호출한 그곳에 특정 값을 반환하게 할 수 있습니다. 이때 이 특정 값을 반환 값(return value)이라고 부릅니다.
인수로 받은 두 값을 더해주는 간단한 함수를 만들어 반환 값에 대해 알아보도록 하겠습니다.
지시자 return
은 함수 내 어디서든 사용할 수 있습니다. 실행 흐름이 지시자 return
을 만나면 함수 실행은 즉시 중단되고 함수를 호출한 곳에 값을 반환합니다. 위 예시에선 반환 값을 result
에 할당하였습니다.
아래와 같이 함수 하나에 여러 개의 return
문이 올 수도 있습니다.
아래와 같이 지시자 return
만 명시하는 것도 가능합니다. 이런 경우는 함수가 즉시 종료됩니다.
예시:
위 예시에서, checkAge(age)
가 false
를 반환하면, (*)
로 표시한 줄은 실행이 안 되기 때문에 함수 showMovie
는 얼럿 창을 보여주지 않습니다.
return
문이 없거나 return
지시자만 있는 함수는 undefined
를 반환합니다.
return
문이 없는 함수도 무언가를 반환합니다. undefined
를 반환하죠.
return
지시자만 있는 경우도 undefined
를 반환합니다. return
은 return undefined
와 동일하게 동작하죠.
return
과 값 사이에 절대 줄을 삽입하지 마세요.
반환하려는 값이 긴 표현식인 경우, 아래와 같이 지시자 return
과 반환하려는 값 사이에 새 줄을 넣어 코드를 작성하고 싶을 수도 있습니다.
자바스크립트는 return문 끝에 세미콜론을 자동으로 넣기 때문에 이렇게 return
문을 작성하면 안 됩니다. 위 코드는 아래 코드처럼 동작합니다.
따라서 반환하고자 했던 표현식을 반환하지 못하고 아무것도 반환하지 않는 것처럼 되어버립니다.
표현식을 여러 줄에 걸쳐 작성하고 싶다면 표현식이 return
지시자가 있는 줄에서 시작하도록 작성해야 합니다. 또는 아래와 같이 여는 괄호를 return
지시자와 같은 줄에 써줘도 괜찮습니다.
이렇게 하면 의도한 대로 표현식을 반환할 수 있습니다.
함수는 어떤 동작
을 수행하기 위한 코드를 모아놓은 것입니다. 따라서 함수의 이름은 대개 동사입니다. 함수 이름은 가능한 한 간결하고 명확해야 합니다. 함수가 어떤 동작을 하는지 설명할 수 있어야 하죠. 코드를 읽는 사람은 함수 이름만 보고도 함수가 어떤 기능을 하는지 힌트를 얻을 수 있어야 합니다.
함수가 어떤 동작을 하는지 축약해서 설명해 주는 동사를 접두어로 붙여 함수 이름을 만드는 게 관습입니다. 다만, 팀 내에서 그 뜻이 반드시 합의된 접두어만 사용해야 합니다.
"show"
로 시작하는 함수는 대개 무언가를 보여주는 함수입니다.
이 외에 아래와 같은 접두어를 사용할 수 있습니다.
"get…"
– 값을 반환함"calc…"
– 무언가를 계산함"create…"
– 무언가를 생성함"check…"
– 무언가를 확인하고 불린값을 반환함
위 접두어를 사용하면 아래와 같은 함수를 만들 수 있습니다.
접두어를 적절히 활용하면 함수 이름만 보고도 함수가 어떤 동작을 하고 어떤 값을 반환하는지 쉽게 알 수 있습니다.
함수는 동작 하나만 담당해야 합니다.
함수는 함수 이름에 언급되어 있는 동작을 정확히 수행해야 합니다. 그 이외의 동작은 수행해선 안 됩니다.
독립적인 두 개의 동작은 독립된 함수 두 개에서 나눠서 수행할 수 있게 해야 합니다. 한 장소에서 두 동작을 동시에 필요로 하는 경우라도 말이죠(이 경우는 제3의 함수를 만들어 그곳에서 두 함수를 호출합니다).
개발자들이 빈번히 하는 실수를 소개해 보겠습니다.
getAge
함수는 나이를 얻어오는 동작만 수행해야 합니다.alert
창에 나이를 출력해 주는 동작은 이 함수에 들어가지 않는 것이 좋습니다.createForm
함수는 form을 만들고 이를 반환하는 동작만 해야 합니다. form을 문서에 추가하는 동작이 해당 함수에 들어가 있으면 좋지 않습니다.checkPermission
함수는 승인 여부를 확인하고 그 결과를 반환하는 동작만 해야 합니다. 승인 여부를 보여주는 메시지를 띄우는 동작이 들어가 있으면 좋지 않습니다.
위 예시들은 접두어의 의미가 합의되었다고 가정하고 만들었습니다. 본인이나 본인이 속한 팀에서 접두어의 의미를 재합의하여 함수를 만들 수도 있긴 하지만, 아마도 위 예시에서 사용한 접두어 의미와 크게 차이가 나진 않을 겁니다. 어찌 되었든 접두어를 사용하여 함수 이름을 지을 땐, 해당 접두어에 어떤 의미가 있는지 잘 이해하고 있어야 합니다. 해당 접두어가 붙은 함수가 어떤 동작을 하는지, 어떤 동작은 하지 못하는지 알고 있어야 하죠. 접두어를 붙여 만든 모든 함수는 팀에서 만든 규칙을 반드시 따라야 합니다. 팀원들은 이 규칙을 충분히 이해하고 있어야 하며, 팀원들 사이에 이 규칙이 잘 공유되어야 합니다.
아주 짧은 이름
정말 빈번히 쓰이는 함수 중에 이름이 아주 짧은 함수가 있습니다.
jQuery 프레임워크에서 쓰이는 함수 $
와 Lodash 라이브러리의 핵심 함수 _
말이죠.
이 함수들은 지금까지 소개한 함수 이름짓기에 관련된 규칙을 지키지 않고 있습니다. 예외에 속하죠. 함수 이름은 간결하고 함수가 어떤 일을 하는지 설명할 수 있게 지어야 합니다.
함수는 간결하고, 한 가지 기능만 수행할 수 있게 만들어야 합니다. 함수가 길어지면 함수를 잘게 쪼갤 때가 되었다는 신호로 받아들이셔야 합니다. 함수를 쪼개는 건 쉬운 작업은 아닙니다. 하지만 함수를 분리해 작성하면 많은 장점이 있기 때문에 함수가 길어질 경우엔 함수를 분리해 작성할 것을 권유합니다.
함수를 간결하게 만들면 테스트와 디버깅이 쉬워집니다. 그리고 함수 그 자체로 주석의 역할까지 합니다!
같은 동작을 하는 함수, showPrimes(n)
를 두 개 만들어 비교해 봅시다. showPrimes(n)
은 n
까지의 소수(prime numbers)를 출력해줍니다.
첫 번째 showPrimes(n)
에선 레이블을 사용해 반복문을 작성해보았습니다.
두 번째 showPrimes(n)
는 소수인지 아닌지 여부를 검증하는 코드를 따로 분리해 isPrime(n)
이라는 함수에 넣어서 작성했습니다.
두 번째 showPrimes(n)
가 더 이해하기 쉽지 않나요? isPrime
함수 이름을 보고 해당 함수가 소수 여부를 검증하는 동작을 한다는 걸 쉽게 알 수 있습니다. 이렇게 이름만 보고도 어떤 동작을 하는지 알 수 있는 코드를 자기 설명적(self-describing) 코드라고 부릅니다.
위와 같이 함수는 중복을 없애려는 용도 외에도 사용할 수 있습니다. 이렇게 함수를 활용하면 코드가 정돈되고 가독성이 높아집니다.
함수 선언 방식으로 함수를 만들 수 있습니다.
함수에 전달된 매개변수는 복사된 후 함수의 지역변수가 됩니다.
함수는 외부 변수에 접근할 수 있습니다. 하지만 함수 바깥에서 함수 내부의 지역변수에 접근하는 건 불가능합니다.
함수는 값을 반환할 수 있습니다. 값을 반환하지 않는 경우는 반환 값이
undefined
가 됩니다.
깔끔하고 이해하기 쉬운 코드를 작성하려면 함수 내부에서 외부 변수를 사용하는 방법 대신 지역 변수와 매개변수를 활용하는 게 좋습니다.
개발자는 매개변수를 받아서 그 변수를 가지고 반환 값을 만들어 내는 함수를 더 쉽게 이해할 수 있습니다. 매개변수 없이 함수 내부에서 외부 변수를 수정해 반환 값을 만들어 내는 함수는 쉽게 이해하기 힘듭니다.
함수 이름을 지을 땐 아래와 같은 규칙을 따르는 것이 좋습니다.
함수 이름은 함수가 어떤 동작을 하는지 설명할 수 있어야 합니다. 이렇게 이름을 지으면 함수 호출 코드만 보아도 해당 함수가 무엇을 하고 어떤 값을 반환할지 바로 알 수 있습니다.
함수는 동작을 수행하기 때문에 이름이 주로 동사입니다.
create…
,show…
,get…
,check…
등의 잘 알려진 접두어를 사용해 이름을 지을 수 있습니다. 접두어를 사용하면 함수 이름만 보고도 해당 함수가 어떤 동작을 하는지 파악할 수 있습니다.
함수는 스크립트를 구성하는 주요 구성 요소입니다. 지금까지 다룬 내용은 함수의 기본입니다. 여기선 함수를 만드는 방법, 사용하는 방법을 소개했는데 이 내용은 시작일 뿐입니다. 이어지는 챕터에선 지금까지 배운 것을 바탕으로 함수가 제공하는 고급 기능에 대해 학습해 보도록 하겠습니다.
Last updated