주요 렌더링 경로 측정

Ilya Grigorik
Ilya Grigorik

게시일: 2014년 3월 31일

모든 견고한 성능 전략의 기초는 훌륭한 측정과 계측입니다. 측정할 수 없는 항목은 최적화할 수 없습니다. 이 가이드에서는 CRP 성능을 측정하는 다양한 접근 방식을 설명합니다.

  • Lighthouse 접근방식에서는 일련의 자동화된 테스트를 페이지에 실행한 다음, 페이지의 CRP 성능에 대한 보고서를 생성합니다. 이 접근 방식은 브라우저에 로드된 특정 페이지의 CRP 성능을 쉽고 빠르게 측정해 주며, 신속하게 테스트를 수행하고 반복하여 성능을 개선해 줍니다.
  • Navigation Timing API 접근방식에서는 RUM(실시간 사용자 모니터링) 측정항목을 캡처합니다. 이름에서 알 수 있듯이 이러한 측정항목은 사용자와 사이트의 실제 상호작용에서 캡처되며 다양한 기기 및 네트워크 조건에서 사용자가 경험하는 실제 CRP 성능을 정확하게 보여줍니다.

일반적으로 좋은 접근 방식은 Lighthouse를 사용하여 명확한 CRP 최적화 기회를 파악한 다음 Navigation Timing API로 코드를 계측하여 앱의 실제 성능을 모니터링하는 것입니다.

Lighthouse는 웹 앱 감사 도구이며 해당 페이지에 대해 일련의 테스트를 수행한 다음, 이 페이지의 결과를 통합된 보고서로 표시해줍니다. Lighthouse를 Chrome 확장 프로그램이나 NPM 모듈로 실행할 수 있으며, 이는 Lighthouse와 지속적 통합 시스템을 통합하는 데 유용합니다.

시작하려면 Lighthouse로 웹 앱 감사를 참고하세요.

Lighthouse를 Chrome 확장 프로그램으로 실행하면 페이지의 CRP 결과가 다음 스크린샷과 같이 표시됩니다.

Lighthouse의 CRP 감사

이 감사의 결과에 대한 자세한 내용은 주요 요청 체인을 참고하세요.

Navigation Timing API와 페이지가 로드될 때 내보낸 기타 브라우저 이벤트를 함께 사용하면 모든 페이지의 실제 CRP 성능을 캡처하고 기록할 수 있습니다.

Navigation Timing

위 다이어그램의 각 라벨은 브라우저가 로드하는 모든 페이지에 대해 추적하는 고해상도 타임스탬프에 해당합니다. 사실상, 이 특정 경우에서는 다양한 모든 타임스탬프 중 일부만 보여줍니다. 지금은 모든 네트워크 관련 타임스탬프를 건너뛰지만 이후 과정에서 이에 대해 다시 살펴볼 것입니다.

그렇다면 이러한 타임스탬프가 의미하는 바는 무엇일까요?

  • domLoading: 전체 프로세스의 시작 타임스탬프입니다. 브라우저가 처음 수신한 HTML 문서 바이트의 파싱을 시작하려고 합니다.
  • domInteractive: 브라우저가 파싱을 완료한 시점을 표시합니다. 모든 HTML 및 DOM 생성이 완료된 상태입니다.
  • domContentLoaded: DOM이 준비되고 자바스크립트 실행을 차단하는 스타일시트가 없는 시점을 표시합니다. 즉, 이제 (잠재적으로) 렌더링 트리를 생성할 수 있습니다.
    • 많은 JavaScript 프레임워크가 자체 로직을 실행하기 전에 이 이벤트를 기다립니다. 이러한 이유로 브라우저는 EventStartEventEnd 타임스탬프를 캡처합니다. 이를 통해 이 실행이 얼마나 오래 걸렸는지 추적할 수 있습니다.
  • domComplete: 이름에서 알 수 있듯이 모든 처리가 완료되고 페이지의 모든 리소스 (이미지 등) 다운로드가 완료되었습니다. 즉, 로드 스피너의 회전이 중단됩니다.
  • loadEvent: 각 페이지 로드의 최종 단계로, 브라우저가 추가 애플리케이션 로직을 트리거할 수 있는 onload 이벤트를 발생시킵니다.

HTML 사양은 각 이벤트에 대한 특정 조건, 즉 이벤트가 실행되는 시점, 충족해야 하는 조건, 기타 중요한 고려사항을 규정합니다. 여기서는 주요 렌더링 경로와 관련된 몇 가지 주요 이정표를 중점적으로 살펴보겠습니다.

  • domInteractive는 DOM이 준비된 시점을 표시합니다.
  • domContentLoaded는 일반적으로 DOM 및 CSSOM이 모두 준비되었음을 표시합니다.
    • JavaScript를 차단하는 파서가 없으면 domInteractive 직후에 DOMContentLoaded가 실행됩니다.
  • domComplete는 페이지 및 모든 하위 리소스가 준비되었음을 표시합니다.
<!DOCTYPE html>
<html>
  <head>
    <title>Critical Path: Measure</title>
    <meta name="viewport" content="width=device-width,initial-scale=1" />
    <link href="style.css" rel="stylesheet" />
    <script>
      function measureCRP() {
        var t = window.performance.timing,
          interactive = t.domInteractive - t.domLoading,
          dcl = t.domContentLoadedEventStart - t.domLoading,
          complete = t.domComplete - t.domLoading;
        var stats = document.createElement('p');
        stats.textContent =
          'interactive: ' +
          interactive +
          'ms, ' +
          'dcl: ' +
          dcl +
          'ms, complete: ' +
          complete +
          'ms';
        document.body.appendChild(stats);
      }
    </script>
  </head>
  <body onload="measureCRP()">
    <p>Hello <span>web performance</span> students!</p>
    <div><img src="awesome-photo.jpg" /></div>
  </body>
</html>

직접 해 보기

위 예시는 처음 보면 조금 복잡해 보일 수 있지만 사실은 매우 간단합니다. Navigation Timing API는 모든 관련 타임스탬프를 캡처하고 개발자 코드는 onload 이벤트가 발생하기를 기다립니다. onload 이벤트는 domInteractive, domContentLoaded, domComplete 이벤트 이후에 발생한다는 사실을 상기하세요. 또한 이 API는 다양한 타임스탬프 간의 차이를 계산합니다.

NavTiming 데모

모든 것이 말한대로 이루어졌으면, 이제 측정할 몇 가지 특정한 마일스톤과 이러한 측정 결과를 출력하기 위한 기본 함수가 만들어졌을 것입니다. 이러한 측정항목을 페이지에 출력하는 대신 코드를 수정하여 이러한 측정항목을 분석 서버로 전송할 수도 있습니다(Google 애널리틱스에서 자동으로 처리). 이는 페이지의 실적을 계속 확인하고 최적화 작업의 이점을 누릴 수 있는 후보 페이지를 식별하는 데 유용한 방법입니다.

DevTools는 어떤가요?

이 문서에서는 CRP 개념을 설명하기 위해 Chrome DevTools Network 패널을 사용하기도 하지만, DevTools에는 중요한 리소스를 분리하기 위한 기본 제공 메커니즘이 없으므로 CRP 측정에 적합하지 않습니다. 이러한 리소스를 식별하도록 도우려면 Lighthouse 감사를 실행하세요.

의견