반응형
Jake Seo
제이크서 위키 블로그
Jake Seo
전체 방문자
오늘
어제
  • 분류 전체보기 (715)
    • 일상, 일기 (0)
    • 백준 문제풀이 (1)
    • 릿코드 문제풀이 (2)
    • 알고리즘 이론 (10)
      • 기본 이론 (2)
      • 배열과 문자열 (8)
    • 데이터베이스 (15)
      • Planet Scale (1)
      • MSSQL (9)
      • 디비 기본 개념 (1)
      • SQLite 직접 만들어보기 (4)
    • 보안 (7)
    • 설계 (1)
    • 네트워크 (17)
      • HTTP (9)
      • OSI Layers (5)
    • 회고 (31)
      • 연간 회고 (2)
      • 주간 회고 (29)
    • 인프라 (52)
      • 도커 (12)
      • AWS (9)
      • 용어 (21)
      • 웹 성능 (1)
      • 대규모 서비스를 지탱하는 기술 (9)
    • 깃 (7)
    • 빌드 도구 (7)
      • 메이븐 (6)
      • 그레이들 (0)
    • Java (135)
      • 이펙티브 자바 (73)
      • 자바 API (4)
      • 자바 잡지식 (30)
      • 자바 디자인 패턴 (21)
      • 톰캣 (Tomcat) (7)
    • 프레임워크 (64)
      • next.js (14)
      • 스프링 프레임워크 (28)
      • 토비의 스프링 (6)
      • 스프링 부트 (3)
      • JPA (Java Persistence API) (5)
      • Nest.js (8)
    • 프론트엔드 (48)
      • 다크모드 (1)
      • 노드 패키지 관리 매니저 (3)
      • CSS (19)
      • Web API (11)
      • tailwind-css (1)
      • React (5)
      • React 새 공식문서 요약 (1)
      • HTML (Markup Language) (5)
    • 자바스크립트 (108)
      • 모던 자바스크립트 (31)
      • 개념 (31)
      • 정규표현식 (5)
      • 코드 스니펫 (1)
      • 라이브러리 (6)
      • 인터뷰 (24)
      • 웹개발자를 위한 자바스크립트의 모든 것 (6)
      • 팁 (2)
    • Typescript (49)
    • 리눅스와 유닉스 (10)
    • Computer Science (1)
      • Compiler (1)
    • IDE (3)
      • VSCODE (1)
      • IntelliJ (2)
    • 세미나 & 컨퍼런스 (1)
    • 용어 (개발용어) (16)
      • 함수형 프로그래밍 용어들 (1)
    • ORM (2)
      • Prisma (2)
    • NODEJS (2)
    • cypress (1)
    • 리액트 네이티브 (React Native) (31)
    • 러스트 (Rust) (15)
    • 코틀린 (Kotlin) (4)
      • 자바에서 코틀린으로 (4)
    • 정규표현식 (3)
    • 구글 애널리틱스 (GA) (1)
    • SEO (2)
    • UML (2)
    • 맛탐험 (2)
    • 리팩토링 (1)
    • 서평 (2)
    • 소프트웨어 공학 (18)
      • 테스팅 (16)
      • 개발 프로세스 (1)
    • 교육학 (1)
    • 삶의 지혜, 통찰 (1)
    • Chat GPT (2)
    • 쉘스크립트 (1)
    • 컴파일 (2)
    • Dart (12)
    • 코드팩토리의 플러터 프로그래밍 (4)
    • 플러터 (17)
    • 안드로이드 스튜디오 (1)
    • 윈도우즈 (1)
    • 잡다한 백엔드 지식 (1)
    • 디자인 패턴 (1)

블로그 메뉴

  • 홈
  • 태그
  • 방명록

공지사항

인기 글

태그

  • 자바 검증
  • serverless computing
  • item8
  • 외래키 제약조건
  • 팩터리 메서드 패턴
  • 프로그래머의 뇌
  • 도커공식문서
  • 러스트
  • MSSQL
  • rust
  • 빈 검증
  • 싱글톤
  • 자바
  • 알고리즘
  • 이펙티브자바
  • 이펙티브 자바 item9
  • 자바스크립트 면접
  • 추상 팩터리 패턴
  • item9
  • prerendering
  • 느린 쿼리
  • 싱글톤 패턴
  • 스프링 검증
  • try-with-resources
  • Pre-rendering
  • Javadoc 자바독 자바주석 주석 Comment
  • 디자인패턴
  • 이펙티브 자바
  • 메이븐 라이프사이클
  • item7
  • 메이븐 페이즈
  • 자바 디자인패턴
  • 메이븐 골
  • 자료구조
  • 슬로우 쿼리
  • 자바스크립트 인터뷰
  • Java
  • next js app
  • 싱글턴
  • pnpm
  • 토비의 스프링
  • 참조 해제
  • 객체복사
  • 자바스크립트
  • 작업기억공간
  • 플라이웨이트패턴
  • NEXT JS
  • Next.js
  • bean Validation
  • 서버리스 컴퓨팅

최근 댓글

최근 글

티스토리

hELLO · Designed By 정상우.
Jake Seo

제이크서 위키 블로그

script 태그의 defer 키워드와 async 키워드 쉽게 알아보기
자바스크립트/개념

script 태그의 defer 키워드와 async 키워드 쉽게 알아보기

2023. 2. 20. 23:29

defer, async 스크립트란?

  • script 태그에 추가될 수 있는 속성이다.
    • HTML 코드를 보면, <script defer src="a.js"></script> 혹은 <script async src="a.js"></script> 와 같은 구문을 볼 수 있다.
  • 역사를 알면, 이러한 속성 값들이 왜 생겼는지 알 수 있다. 역사를 살펴보며 천천히 배워보자.

script 태그의 특징

  • 초기의 브라우저 자바스크립트는 아주 짧고 간단한 스크립트 코드를 넣기 위해 존재했다. 그런데 동적인 웹의 발달로 자바스크립트 코드는 아주 길게 변했다.
  • 브라우저는 DOM 을 생성하다가도 script 태그를 만나면, script 태그의 내용부터 해석하는 방식으로 동작했다.

script 태그의 특징 때문에 생긴 문제

  • 초기에 의도한대로 head 태그에 자바스크립트 코드를 몰아넣으니 DOM 을 생성하기도 전에 자바스크립트를 해석하느라 페이지 로딩 속도가 매우 느려지는 현상이 일어났다.
  • 자바스크립트 코드에서 head 아래에 있는 DOM 요소를 제어하는 행위도 타이밍이 맞지 않아서 불가능해졌다.

임시방편 해결책: body 태그 끝에 script 를 위치시키기

<html>
  <head>
    <!-- CSS, 메타정보, 타이틀 등 -->
  </head>
  <body>
    <!-- DOM 요소 -->
    <script></script>
  </body>
</html>
  • 더이상 DOM 로드를 막지 않는다.
  • 다만 여전히 아주 큰 HTML 페이지에 대해서는 많은 지연이 발생한다.
    • 특히 상대적으로 통신이 느린 모바일 환경에서 더 부각될 것이다.

defer 의 등장

  • script 태그에 defer 속성을 붙이면 백그라운드에서 스크립트를 다운받는다.
    • 백그라운드에서 돌기 때문에 스크립트 다운로드 중에도 HTML 파싱이 멈추지 않는다.
  • 페이지 컨텐츠는 즉시 출력된다.
  • defer 속성이 추가된 스크립트는 DOM 이 준비된 이후에 실행된다.
    • 하지만 DOMContnetLoaded 이벤트가 발생되기 전에 실행된다.
<html>
  <head>
    <!-- CSS, 메타정보, 타이틀 등 -->
    <script defer src="a.js"></script>
    <script>
      document.addEventListener("DOMContentLoaded", () =>
        console.log("`defer` 스크립트가 실행된 후 실행된다.")
      );
    </script>
  </head>
  <body>
    <!-- DOM 요소 -->
  </body>
</html>

defer 의 특징

  • 작은 스크립트가 먼저 다운로드 될 수 있지만, 실행은 순서대로 된다.
    • 스크립트는 병렬적으로 다운로드 되기 때문에 작은 스크립트가 먼저 다운로드 완료될 수 있다.
    • 다만, 스크립트 실행 순서는 문서에 추가된대로 되어야 한다고 정의되어 있기 때문에 실행은 추가된 순서대로 실행된다.

defer 가 동작하지 않는 경우

  • 인라인 스크립트로 이루어진 경우, defer 가 동작하지 않는다.
<script defer>
  document.getElementById("hello").innerHTML = "HELLO!";
</script>

async 의 등장

  • defer 와 마찬가지로 백그라운드에서 다운로드된다.
  • async 는 완벽히 독립적으로 동작한다.
    • DOMContentLoaded 전, 후 아무 때나 실행될 수 있다.
    • 스크립트들 끼리도 실행 순서가 보장되지 않는다.
  • async 실행 중에는 HTML 파싱이 멈추게 된다.
<html>
  <head>
    <!-- CSS, 메타정보, 타이틀 등 -->
    <script async src="a.js"></script>
    <script>
      document.addEventListener("DOMContentLoaded", () =>
        console.log("실행 순서가 보장되지 않는다.")
      );
    </script>
  </head>
  <body>
    <!-- DOM 요소 -->
  </body>
</html>

async 의 특징

  • 실행 순서의 보장 없이 먼저 로드된 것이 먼저 실행된다.
    • 이를 load-first order 라고 한다.
  • DOM 에 전혀 영향을 받지 않는 서드파티 스크립트를 삽입할 때 아주 유용하다.
    • google analytics 같은 것이 좋은 예다.

동적 자바스크립트

  • 약간 괴기한 문법이 있다.
  • 보통 async 와 동일한 특징을 갖는다.
    • load-first order 를 갖는다.
let script = document.createElement("script");
script.src = "/a.js";
document.body.append(script);

async = false

  • script.async = false 로 설정해주면, 추가한 순서대로 실행된다.
  • a.js 후에 b.js 가 실행될 것이다.
function loadJS(src) {
  let script = document.createElement("script");
  script.src = src;
  script.async = false;
  document.body.append(script);
}

loadJS("/a.js");
loadJS("/b.js");

async 와 defer 비교

async

  • 백그라운드에서 다운받아진다.
  • load-first order 로 순서 없이 먼저 로드된 것이 먼저 실행된다.
  • DOMContentLoaded 를 신경쓰지 않는다.

쓰임새: DOM 이나 실행 순서에 연관 없는 스크립트
ex) 방문자 수 카운터, 광고 관련 스크립트

defer

  • 백그라운드에서 다운받아진다.
  • 문서에 추가된 순서대로 실행된다.
  • DOM 이 로딩을 끝낸 후에 실행된다.

쓰임새: DOM 이 로드된 이후에 순서대로 실행되어야 하는 스크립트
ex) UI 렌더 스크립트

그림으로 살펴보기

type="module" 이 속성에 추가되면?

  • 타임라인이 조금 달라지는데, 아무것도 안붙이면 defer 와 동일한 타임라인을 갖는다.
    • defer 와 다르게 src 속성을 이용한 스크립트가 아닌 인라인 스크립트 (<script></script>) 에도 적용이 된다.
  • 추가적으로 async 를 붙이면, async 와 매우 흡사하게 동작한다.
반응형
저작자표시 비영리 (새창열림)

'자바스크립트 > 개념' 카테고리의 다른 글

자바스크립트 이벤트(Event) 객체와 커스텀 이벤트 (Custom Event) 란?  (0) 2023.07.07
자바스크립트가 문자열을 표현하는 방식 (feat.UTF-16)  (0) 2023.03.06
매크로 태스크 (Macro Task) 와 마이크로 태스크 (Micro Task) 란?  (0) 2023.02.13
자바스크립트의 실행 컨텍스트 (Execution Context) 란?  (0) 2023.02.05
Lexical Scope (렉시컬 스코프) 란? (자바스크립트의 스코핑 방식)  (0) 2023.02.02
    '자바스크립트/개념' 카테고리의 다른 글
    • 자바스크립트 이벤트(Event) 객체와 커스텀 이벤트 (Custom Event) 란?
    • 자바스크립트가 문자열을 표현하는 방식 (feat.UTF-16)
    • 매크로 태스크 (Macro Task) 와 마이크로 태스크 (Micro Task) 란?
    • 자바스크립트의 실행 컨텍스트 (Execution Context) 란?
    Jake Seo
    Jake Seo
    ✔ 잘 보셨다면 광고 한번 클릭해주시면 큰 힘이 됩니다. ✔ 댓글로 틀린 부분을 지적해주시면 기분 나빠하지 않고 수정합니다. ✔ 많은 퇴고를 거친 글이 좋은 글이 된다고 생각합니다. ✔ 간결하고 명료하게 사람들을 이해 시키는 것을 목표로 합니다.

    티스토리툴바