...
IIFE 이란?
"Immediately Invoked Function Expression"의 줄임말로, 정의 되자마자 즉시 실행되는 함수 표현식을 말한다.
// 아래 두 함수는 동일한 동작을 수행한다.
var app = function() {
console.log('함수 호출'); // "함수 호출" 출력
};
app();
// 즉시실행함수 IIFE
(function() {
console.log('함수 호출'); // "함수 호출" 출력
}());
위의 두 함수는 모두 동일한 로직을 가지고 있다. IIFE는 전체 익명함수를 괄호로 감싸줌으로써 내부 코드가 선언문이 아니라 표현식인 것처럼 Parser를 속인다. 단, 익명함수이기 때문에 재사용은 적합하지 않다. 물론 기명으로도 가능은 하다. 하지만 보통 함수에게 이름을 짓는 것은 호출이 목적인 경우가 대부분인데, 한번 자동으로 실행된 이후 생명을 다하는 IIFE에게 이름을 지어주는 것은 의미가 없다. 어찌되었든 IIFE를 익명으로써야하는지, 기명으로 써야하는지에 대해서는 개발자들 사이에서도 의견이 갈리는 것 같다.
즉시실행 함수 표현식 문법 종류
아래는 JavaScript에서 사용되는 즉시 실행 함수 표현식의 3가지 문법을 보여준다.
// 아래 함수선언문은 아무것도 반환하지 않는다(혹은 undefined)
function() {
}
논리 부정 연산자(!)
논리 부정 연산자(!)를 사용해서 즉시 실행 함수를 구현할 수 있다. 이렇게 하면 함수는 undefined를 반환하며, 논리 부정 연산자에 의해 true 값을 되기 때문에 즉시 실행이 되는 원리이다.
// 이런 표현식을 쓰면 !undefined는 true를 반환한다. 그리고 바로 실행하기도 함. 단, 리턴값이 없어야함
!function() {
console.log("This code runs immediately.");
}();
괄호 안에서 괄호 문법
함수 선언문을 괄호 기호로 둘러싸고 함수를 호출하는 괄호를 붙이면 즉시 실행 함수로 변환된다. (함수)();
(function() {
console.log('IIFE syntax 1');
})();
Douglas 표기법
이는 Douglas Crockford가 권장하는 즉시 실행 함수 표현식의 표준 표기법이다. 위의 괄호 안에서 괄호 문법과의 차이점은 닫는 괄호 부분을 어디에 두느냐의 차이이다.
(function() {
console.log('IIFE syntax 2');
}());
단, 일반적으로 JavaScript 커뮤니티에서는 더글라스 표기법 보단 괄호 안에서 괄호 문법이 더 선호되며, JSLint와 같은 일부 도구에서는 더글라스 표기법법을 사용할 때 경고를 발생시키도 한다. 그러나 둘 다 괜찮은 방법이며, 코드 작성 스타일에 따라 선택하면 된다.
즉시실행함수를 쓰는 이유
1. 클로저 구현할때
바로 클로저 변수가 적용된 클로저 함수를 얻고 싶을때 쏠쏠히 사용된다.
let func = (function() {
var counter = 0;
return function(){ // 클로저
return ++counter;
}
})();
func(); // 1
func(); // 2
func(); // 3
2. async / await 비동기 처리를 바로 사용할때
await 키워드는 반드시 async function 내에서 쓸수 있다는 제한 때문에 반드시 함수를 정의하고 함수를 호출하는 형식으로 사용하여야 한다. 하지만 모든 코드를 함수로 묶고 호출하는 방식은 가독성 측면에서 좋지 않을 수 있다. 따라서 만일 이를 별도의 함수 호출이 아닌 일반 코드 실행과 같이 즉시 실행을 하고 싶은 경우 다음과 같이 async function을 정의하고 즉시 호출하기도 한다. 이는 정말 자주 사용되는 기법 이기도 하다.
(async () => {
... 코드1
await promise();
... 코드2
})();
다만 최근에 추가된 Top Level Await 기능 덕분에 굳이 async 즉시 실행 함수 문법을 쓰지 않고도 await 키워드를 사용할 수 있게 되었다.
3. 전역 변수 충돌을 방지하고 싶을때
즉시 실행 함수의 진정한 장점은 스코프를 제한하고 전역 변수 충돌을 방지하는 데 사용되어 모듈 패턴을 구현하는 데 있다. IIFE를 사용하면 함수 내에서 변수(지역 변수)를 정의하는것과 같아, 이러한 변수는 IIFE 내에서만 유효하게 된다. 따라서 IIFE를 사용하면 전역 스코프에서 변수 충돌 문제가 발생하지 않게 된다.
(function() {
var x = 10;
console.log(x); // Output: 10
})();
console.log(x); // Output: ReferenceError: x is not defined
위의 예제 코드를 보면 var 키워드로 변수를 선언했지만 전역 변수 접근이 안되는걸 볼 수 있다. 이처럼 IIFE는 전역 스코프에 불필요한 변수를 추가해서 오염시키는 것을 방지할 수 있을 뿐 아니라 IIFE 내부안으로 다른 변수들이 접근하는 것을 막을 수도 있다. 그래서 일회성 변수가 필요하면 전역 변수로 선언하지 않고 안전하게 지역 변수로 다루어 해결이 가능해진다.
전역 변수 뿐만 아니라 전역 함수 충돌 역시 방지할 수도 있다. 예를 들어 IIFE를 사용하여 전역 스코프에서 함수를 정의하면 전역 스코프에서 함수 이름 충돌 문제를 방지할 수 있다.
// sayHello는 오로지 IIFE 블록 내에서만 사용이 가능하다
(function() {
function sayHello(name) {
console.log('Hello, ' + name + '!');
}
sayHello('John');
})();
IIFE를 사용하여 전역 스코프 충돌을 방지하는 것은 JavaScript 개발에서 일반적인 패턴 중 하나이다. 이 패턴은 모듈 패턴과 유사해서 자바스크립트 모듈의 시초이기도 기본 뼈대 이기도 하다.
IIFE 와 세미콜론 주의점
일반적으로 자바스크립트 코드에서는 세미콜론(;) 을 생략해도 실행하는데 문제가 없다. 하지만 모든 경우에서 생략 가능한 것은 아니다. 특히, IIFE 와 같이 사용할때에는 세미콜론을 생략하지 말아야 한다. 그렇지 않으면 아래와 같이 에러가 뜨기 때문이다.
const a = 1 // 세미콜론 생략함
(function () {
console.log(1)
})()
위 코드를 보면, const a = 1 에 세미콜론을 붙이지 않았음을 볼 수 있다. 일반적으로 const a = 1 에 세미콜론을 안붙여다고 해서 변수 a 에 1이 할당되지 않은 것은 아니다. 문제는 그 다음에 이어지는 즉시 실행 함수 때문에 에러가 나는 것이다.
세미콜론이 없으면 자바스크립트 엔진은 다음 줄의 코드를 이전 줄과 연결하여 해석하려고 시도하게 된다. 그래서 이 경우에는 const a = 1 다음에 오는 (function () {console.log(1)})() 즉시 실행 함수가 함수 호출을 의미하는 것이 아니라 함수 표현식을 감싼 괄호로 인식되게 된다. 따라서 위의 코드는 const a = 1 곱하기 (function () {console.log(1)})() 로 해석되고, 이는 유효한 문법이 아니므로 에러가 발생하게 된다.
자바스크립트 괄호 해석 원리
갑자기 곱셈이 왜 나오냐 하면, 자바스크립트에서는 괄호가 여러 가지 의미를 가질 수 있기 때문이다. 예를 들어, 괄호는 함수 호출을 의미할 수도 있고, 함수 표현식을 감싸는 역할을 할 수도 있고, 연산자 우선순위를 조절하는 역할을 할 수도 있다. 그런데 만약 괄호 앞에 세미콜론이 없다면, 자바스크립트 엔진은 괄호를 함수 호출이 아니라 연산자 우선순위를 조절하는 역할로 해석해 버린다.
예를 들어, 2 * (3 + 4) 라는 코드가 있다면, 자바스크립트 엔진은 괄호 안의 3 + 4 를 먼저 계산하고 그 결과에 2 를 곱하는 방식으로 해석합니다. 이때 괄호는 연산자 우선순위를 조절하는 역할을 한다. 그런데 만약 const a = 1 (function () {console.log(1)})() 와 같은 세미콜론이 생략된 코드가 있다면, 자바스크립트 엔진은 괄호를 함수 호출이 아니라 연산자 우선순위를 조절하는 역할로 해석하게 된다. 그래서 const a = 1 곱하기 (function () {console.log(1)})() 로 해석하고, 이는 유효한 문법이 아니므로 에러가 발생되는 것이다.
좀 더 이해하기 편하게 변수가 아닌 함수 선언식으로 비교해보자. 변수 x 는 엄연히 함수이다. 하지만 이 역시 실행하면 똑같은 에러가 발생하게 된다.
var x = function () {
}
(function () {
console.log('hello')
})()
이 역시 자바스크립트 엔진은 IIFE 를 앞의 코드와 연결하여 해석하기 때문에 var x = function () {} 곱하기 (function () {console.log('hello')})() 로 해석하고 에러를 발생시킨 것이다. 이처럼 세미콜론의 유무에 따라 괄호의 의미가 달라질 수 있으므로 주의해야 한다.
세미콜론은 왠만하면 다 붙이자
따라서 최종 정리하자면 IIFE 와 일반 코드를 연달아 사용할때는 IIFE 앞에 세미콜론을 붙여주면 이런 문제를 방지할 수 있다.
그런데 이러한 복잡한 원리를 항상 머릿속에 인지하면서 프로그래밍을 하는 것보다, 그냥 여타 C언어와 자바 언어와 같이 반드시 한 라인의 코드가 끝나면 세미콜론을 붙여야 하는 것 처럼, 그냥 자바스크립트도 코딩 할때 세미콜론을 항상 마지막에 붙여주는 습관을 들이면 이러한 웃기는 현상을 피해갈 수 있을 것이다.
중괄호 뒤에 세미콜론을 붙이는건 C언어나 자바에서도 없는 좀 괴상한 패턴 같지만, 자바스크립트 언어에서는 실행하면 코드에 자동 세미콜론(Automatic Semicolon Insertion) 처리가 되기 때문에 안티 코드 패턴은 아니다. 이러한 특징 때문에 자바스크립트 개발자들은 의도치 않은 결과를 방지하기 위해 일부로라도 붙여주는 편이다.
이 글이 좋으셨다면 구독 & 좋아요
여러분의 구독과 좋아요는
저자에게 큰 힘이 됩니다.