    <header>
      <h1 class="page-title">
        Frontend Timing: Page Loading, Browser Events, rAF, and Component
        Lifecycle
      </h1>
      <p class="page-description"></p>
      <table class="properties">
        <tbody>
          <tr class="property-row property-row-created_by">
            <th>
              <span class="icon property-icon"
                ><div
                  data-testid="/icons/user-circle_gray.svg"
                  style="
                    width: 14px;
                    height: 14px;
                    flex-shrink: 0;
                    transform: scale(1.2);
                    mask: url(/icons/user-circle_gray.svg?mode=light) no-repeat
                      center;
                    -webkit-mask: url(/icons/user-circle_gray.svg?mode=light)
                      no-repeat center;
                    background-color: rgba(142, 139, 134, 1);
                    fill: rgba(142, 139, 134, 1);
                  "
                ></div></span
              >Created by
            </th>
            <td>
              <span class="user"
                ><img
                  src="Frontend%20Timing%20Page%20Loading,%20Browser%20Events,%20rAF,%202756cd51990d807abf73cfdbb6be0e29/Screenshot_2025-08-02_at_10.38.11_AM.png"
                  class="icon user-icon"
                />JiaLin Huang</span
              >
            </td>
          </tr>
          <tr class="property-row property-row-last_edited_time">
            <th>
              <span class="icon property-icon"
                ><div
                  data-testid="/icons/clock_gray.svg"
                  style="
                    width: 14px;
                    height: 14px;
                    flex-shrink: 0;
                    transform: scale(1.2);
                    mask: url(/icons/clock_gray.svg?mode=light) no-repeat center;
                    -webkit-mask: url(/icons/clock_gray.svg?mode=light)
                      no-repeat center;
                    background-color: rgba(142, 139, 134, 1);
                    fill: rgba(142, 139, 134, 1);
                  "
                ></div></span
              >Last edited
            </th>
            <td><time>@2025年9月21日 19:08</time></td>
          </tr>
          <tr class="property-row property-row-multi_select">
            <th>
              <span class="icon property-icon"
                ><div
                  data-testid="/icons/list_gray.svg"
                  style="
                    width: 14px;
                    height: 14px;
                    flex-shrink: 0;
                    transform: scale(1.2);
                    mask: url(/icons/list_gray.svg?mode=light) no-repeat center;
                    -webkit-mask: url(/icons/list_gray.svg?mode=light) no-repeat
                      center;
                    background-color: rgba(142, 139, 134, 1);
                    fill: rgba(142, 139, 134, 1);
                  "
                ></div></span
              >Tags
            </th>
            <td>
              <span class="selected-value select-value-color-purple">Post</span
              ><span class="selected-value select-value-color-blue"
                >Reactjs</span
              ><span class="selected-value select-value-color-pink">Vue</span>
            </td>
          </tr>
        </tbody>
      </table>
    </header>
    <div class="page-body">
      <h1 class="">Root cause</h1>
      <p class="">
        When using location.href to jump to another page in Vue2&#x27;s
        mounted() hook, Firefox/Safari would get stuck in an endless reload loop
        instead of going to the new page.
      </p>
      <p class="">Test results:</p>
      <ul class="bulleted-list">
        <li style="list-style-type: disc">setTimeout(0) → Failed</li>
      </ul>
      <ul class="bulleted-list">
        <li style="list-style-type: disc">requestAnimationFrame → Failed</li>
      </ul>
      <ul class="bulleted-list">
        <li style="list-style-type: disc">setTimeout(50ms, 100ms) → Failed</li>
      </ul>
      <ul class="bulleted-list">
        <li style="list-style-type: disc">...</li>
      </ul>
      <ul class="bulleted-list">
        <li style="list-style-type: disc">setTimeout(500ms) → Success</li>
      </ul>
      <p class="">
        <mark class="highlight-red"
          ><strong
            >Key finding: mounted timing ≠ page ready for navigation</strong
          ></mark
        >
      </p>
      <p class=""></p>
      <p class="">
        Next, we&#x27;ll explore what render and mount really mean, the
        browser&#x27;s Critical Rendering Path, when requestAnimationFrame runs,
        and the differences between various &quot;complete&quot; states. Once
        you understand these concepts, you&#x27;ll know exactly when to run what
        code and avoid common timing traps.
      </p>
      <p class=""></p>
      <p class=""></p>
      <p class=""></p>
      <p class=""></p>
      <h1 class="">What is Rendering? What Does Render Complete Mean?</h1>
      <p class="">
        <mark class="highlight-red"
          ><strong
            >Core problem: Different roles understand &quot;render
            complete&quot; differently. Pick the wrong timing and you&#x27;ll
            hit bugs.</strong
          ></mark
        >
      </p>
      <h3 class="">Vue/React Developers: Logical Render</h3>
      <p class="">
        For frameworks, render = Virtual DOM calculation. Pure logic, no visual
        output.
      </p>
      <script
        src="https://cdnjs.cloudflare.com/ajax/libs/prism/1.29.0/prism.min.js"
        integrity="sha512-7Z9J3l1+EYfeaPKcGXu3MS/7T+w19WtKQY/n+xzmw4hZhJ9tyYmcUS+4QqAlzhicE5LAfMQSF3iFTK9bQdTxXg=="
        crossorigin="anonymous"
        referrerpolicy="no-referrer"
      ></script>
      <link
        rel="stylesheet"
        href="https://cdnjs.cloudflare.com/ajax/libs/prism/1.29.0/themes/prism.min.css"
        integrity="sha512-tN7Ec6zAFaVSG3TpNAKtk4DOHNpSwKHxxrsiw4GHKESGPs5njn/0sMCUMl2svV4wo4BK/rCP7juYz+zx+l6oeQ=="
        crossorigin="anonymous"
        referrerpolicy="no-referrer"
      />
      <pre
        class="code code-wrap"
      ><code class="language-JavaScript" style="white-space:pre-wrap;word-break:break-all">// React render phase - purely logical
function MyComponent() {
  return &lt;div&gt;Hello World&lt;/div&gt;; // Virtual DOM created, nothing visual yet
}</code></pre>
      <p class="">
        Vue3 uses Block Tree to track only dynamic nodes. React 18 uses time
        slicing to make rendering interruptible:
      </p>
      <script
        src="https://cdnjs.cloudflare.com/ajax/libs/prism/1.29.0/prism.min.js"
        integrity="sha512-7Z9J3l1+EYfeaPKcGXu3MS/7T+w19WtKQY/n+xzmw4hZhJ9tyYmcUS+4QqAlzhicE5LAfMQSF3iFTK9bQdTxXg=="
        crossorigin="anonymous"
        referrerpolicy="no-referrer"
      ></script>
      <link
        rel="stylesheet"
        href="https://cdnjs.cloudflare.com/ajax/libs/prism/1.29.0/themes/prism.min.css"
        integrity="sha512-tN7Ec6zAFaVSG3TpNAKtk4DOHNpSwKHxxrsiw4GHKESGPs5njn/0sMCUMl2svV4wo4BK/rCP7juYz+zx+l6oeQ=="
        crossorigin="anonymous"
        referrerpolicy="no-referrer"
      />
      <pre
        class="code code-wrap"
      ><code class="language-JavaScript" style="white-space:pre-wrap;word-break:break-all">function App() {
  const [query, setQuery] = useState(&#x27;&#x27;);
  const [list, setList] = useState(largeData);
  const [isPending, startTransition] = useTransition();

  const handleSearch = (value) =&gt; {
    setQuery(value); // High priority: update input right away

    startTransition(() =&gt; {
      setList(filterData(value)); // Low priority: can be interrupted
    });
  };
}
</code></pre>
      <h3 class="">Browser: Visual Render</h3>
      <p class="">
        Browser render = complete Style → Layout → Paint → Composite flow.
      </p>
      <p class="">
        CSS calculation → element positioning → pixel drawing → layer
        composition = user sees content
      </p>
      <h3 class="">User: Real Experience</h3>
      <p class="">
        User&#x27;s view of render = &quot;I can see the page and click
        things.&quot;
      </p>
      <table class="simple-table">
        <thead class="simple-table-header">
          <tr>
            <th class="simple-table-header-color simple-table-header">Type</th>
            <th class="simple-table-header-color simple-table-header">
              Complete Standard
            </th>
            <th class="simple-table-header-color simple-table-header">
              Visual Output
            </th>
          </tr>
        </thead>
        <tbody>
          <tr>
            <td class="">Vue/React render</td>
            <td class="">Virtual DOM calculation done</td>
            <td class="">❌ None</td>
          </tr>
          <tr>
            <td class="">Browser render</td>
            <td class="">Paint + Composite done</td>
            <td class="">✅ Yes</td>
          </tr>
          <tr>
            <td class="">User perception</td>
            <td class="">Visible + interactive</td>
            <td class="">✅ Yes</td>
          </tr>
        </tbody>
      </table>
      <p class="">
        This is why running location.href right after mounted() fails: The page
        is logically ready, but not visually and navigationally ready.
      </p>
      <p class=""></p>
      <p class=""></p>
      <p class=""></p>
      <p class=""></p>
      <p class=""></p>
      <p class=""></p>
      <p class=""></p>
      <p class=""></p>
      <p class=""></p>
      <h1 class="">What is Mount</h1>
      <p class="">
        <mark class="highlight-red"
          ><strong
            >Framework &quot;mounted&quot; ≠ user sees the page.</strong
          ></mark
        >
      </p>
      <h3 class="">Framework View: DOM Insertion</h3>
      <p class="">
        mount = Virtual DOM turns into real DOM and gets added to the document.
      </p>
      <script
        src="https://cdnjs.cloudflare.com/ajax/libs/prism/1.29.0/prism.min.js"
        integrity="sha512-7Z9J3l1+EYfeaPKcGXu3MS/7T+w19WtKQY/n+xzmw4hZhJ9tyYmcUS+4QqAlzhicE5LAfMQSF3iFTK9bQdTxXg=="
        crossorigin="anonymous"
        referrerpolicy="no-referrer"
      ></script>
      <link
        rel="stylesheet"
        href="https://cdnjs.cloudflare.com/ajax/libs/prism/1.29.0/themes/prism.min.css"
        integrity="sha512-tN7Ec6zAFaVSG3TpNAKtk4DOHNpSwKHxxrsiw4GHKESGPs5njn/0sMCUMl2svV4wo4BK/rCP7juYz+zx+l6oeQ=="
        crossorigin="anonymous"
        referrerpolicy="no-referrer"
      />
      <pre
        class="code code-wrap"
      ><code class="language-JavaScript" style="white-space:pre-wrap;word-break:break-all">// Vue
onMounted(() =&gt; {
  console.log(&#x27;Component is mounted to DOM&#x27;);
});

// React
useEffect(() =&gt; {
  console.log(&#x27;Component is mounted to DOM&#x27;);
}, []);</code></pre>
      <h3 class="">Browser View: DOM Operation</h3>
      <p class="">
        mount = running appendChild or innerHTML, triggering DOM change
        detection.
      </p>
      <script
        src="https://cdnjs.cloudflare.com/ajax/libs/prism/1.29.0/prism.min.js"
        integrity="sha512-7Z9J3l1+EYfeaPKcGXu3MS/7T+w19WtKQY/n+xzmw4hZhJ9tyYmcUS+4QqAlzhicE5LAfMQSF3iFTK9bQdTxXg=="
        crossorigin="anonymous"
        referrerpolicy="no-referrer"
      ></script>
      <link
        rel="stylesheet"
        href="https://cdnjs.cloudflare.com/ajax/libs/prism/1.29.0/themes/prism.min.css"
        integrity="sha512-tN7Ec6zAFaVSG3TpNAKtk4DOHNpSwKHxxrsiw4GHKESGPs5njn/0sMCUMl2svV4wo4BK/rCP7juYz+zx+l6oeQ=="
        crossorigin="anonymous"
        referrerpolicy="no-referrer"
      />
      <pre
        class="code code-wrap"
      ><code class="language-JavaScript" style="white-space:pre-wrap;word-break:break-all">parentElement.appendChild(newElement); // This is &quot;mounting&quot;</code></pre>
      <h3 class="">User View: Screen Shows Up</h3>
      <p class="">
        mount = &quot;I can see content&quot; = browser finished Paint +
        Composite.
      </p>
      <h3 class="">Problems from Timing Differences</h3>
      <script
        src="https://cdnjs.cloudflare.com/ajax/libs/prism/1.29.0/prism.min.js"
        integrity="sha512-7Z9J3l1+EYfeaPKcGXu3MS/7T+w19WtKQY/n+xzmw4hZhJ9tyYmcUS+4QqAlzhicE5LAfMQSF3iFTK9bQdTxXg=="
        crossorigin="anonymous"
        referrerpolicy="no-referrer"
      ></script>
      <link
        rel="stylesheet"
        href="https://cdnjs.cloudflare.com/ajax/libs/prism/1.29.0/themes/prism.min.css"
        integrity="sha512-tN7Ec6zAFaVSG3TpNAKtk4DOHNpSwKHxxrsiw4GHKESGPs5njn/0sMCUMl2svV4wo4BK/rCP7juYz+zx+l6oeQ=="
        crossorigin="anonymous"
        referrerpolicy="no-referrer"
      />
      <pre
        class="code code-wrap"
      ><code class="language-JavaScript" style="white-space:pre-wrap;word-break:break-all">onMounted(() =&gt; {
  // ❌ This might fail because layout hasn&#x27;t happened yet
  element.scrollIntoView();

  // ✅ Better approach: wait for next frame
  requestAnimationFrame(() =&gt; {
    element.scrollIntoView();
  });
});</code></pre>
      <p class="">
        From mounted to actual screen display, there can be a 16-100ms delay. If
        you need to work with visual elements right away (scrolling, animations,
        measuring sizes), you must wait for the browser to finish rendering.
      </p>
      <p class=""></p>
      <p class=""></p>
      <p class=""></p>
      <p class=""></p>
      <p class=""></p>
      <p class=""></p>
      <h1 class="">Browser Basics: Process, Thread, Pipeline</h1>
      <h3 class="">Browser&#x27;s Multi-Process</h3>
      <p class="">
        Modern browsers aren&#x27;t just one program. They&#x27;re built from
        many separate processes working together. This design makes browsers
        more stable and secure.
      </p>
      <p class="">Main parts:</p>
      <ul class="bulleted-list">
        <li style="list-style-type: disc">
          Browser Process: The control center. Handles your bookmarks, history,
          and user interface
        </li>
      </ul>
      <ul class="bulleted-list">
        <li style="list-style-type: disc">
          Network Process: Takes care of all internet requests and server
          communication
        </li>
      </ul>
      <ul class="bulleted-list">
        <li style="list-style-type: disc">
          GPU Process: Handles graphics and hardware acceleration
        </li>
      </ul>
      <ul class="bulleted-list">
        <li style="list-style-type: disc">
          Renderer Process: Each tab gets its own rendering process
        </li>
      </ul>
      <ul class="toggle">
        <li>
          <details open="">
            <summary>
              <strong>Process and thread tree structure (expand)</strong>
            </summary>
            <script
              src="https://cdnjs.cloudflare.com/ajax/libs/prism/1.29.0/prism.min.js"
              integrity="sha512-7Z9J3l1+EYfeaPKcGXu3MS/7T+w19WtKQY/n+xzmw4hZhJ9tyYmcUS+4QqAlzhicE5LAfMQSF3iFTK9bQdTxXg=="
              crossorigin="anonymous"
              referrerpolicy="no-referrer"
            ></script>
            <link
              rel="stylesheet"
              href="https://cdnjs.cloudflare.com/ajax/libs/prism/1.29.0/themes/prism.min.css"
              integrity="sha512-tN7Ec6zAFaVSG3TpNAKtk4DOHNpSwKHxxrsiw4GHKESGPs5njn/0sMCUMl2svV4wo4BK/rCP7juYz+zx+l6oeQ=="
              crossorigin="anonymous"
              referrerpolicy="no-referrer"
            />
            <pre
              class="code code-wrap"
            ><code class="language-Plain Text" style="white-space:pre-wrap;word-break:break-all">  Browser Application
  ├── Browser Process
  ├── Network Process
  ├── GPU Process
  └── Renderer Process (one per tab)
      ├── Main Thread
      ├── Compositor Thread
      └── Web Worker Threads
</code></pre>
          </details>
        </li>
      </ul>
      <p class=""></p>
      <p class="">
        Why separate tabs work well: When one tab crashes or uses too much
        memory, your other tabs keep working fine. That&#x27;s why you can watch
        a video in one tab while coding in another.
      </p>
      <p class=""></p>
      <h3 class="">What JavaScript Single-Thread Really Means</h3>
      <p class="">
        People say JavaScript is single-threaded, but the browser itself is much
        more complex. JavaScript acts like a &quot;manager&quot; rather than a
        &quot;worker&quot; in this system. When you write
        <mark class="highlight-red">element.style.color = &#x27;red&#x27;</mark
        >, JavaScript just marks that element for update. The actual style
        calculations, repainting, and compositing get handed off to specialized
        threads.
      </p>
      <script
        src="https://cdnjs.cloudflare.com/ajax/libs/prism/1.29.0/prism.min.js"
        integrity="sha512-7Z9J3l1+EYfeaPKcGXu3MS/7T+w19WtKQY/n+xzmw4hZhJ9tyYmcUS+4QqAlzhicE5LAfMQSF3iFTK9bQdTxXg=="
        crossorigin="anonymous"
        referrerpolicy="no-referrer"
      ></script>
      <link
        rel="stylesheet"
        href="https://cdnjs.cloudflare.com/ajax/libs/prism/1.29.0/themes/prism.min.css"
        integrity="sha512-tN7Ec6zAFaVSG3TpNAKtk4DOHNpSwKHxxrsiw4GHKESGPs5njn/0sMCUMl2svV4wo4BK/rCP7juYz+zx+l6oeQ=="
        crossorigin="anonymous"
        referrerpolicy="no-referrer"
      />
      <pre
        class="code code-wrap"
      ><code class="language-JavaScript" style="white-space:pre-wrap;word-break:break-all">// JavaScript only gives orders
element.style.color = &#x27;red&#x27;;        // Main Thread: mark for update
// Other threads do the actual work:
// - Style Thread: recalculate styles
// - Layout Thread: recalculate positions
// - Paint Thread: redraw pixels
// - Compositor Thread: composite to screen</code></pre>
      <p class="">
        Web Workers are different: They&#x27;re the only JavaScript environment
        with their own execution thread, but they can&#x27;t touch the DOM
        directly. They must talk to the main thread through message passing.
      </p>
      <p class=""></p>
      <h3 class="">Thread Division and Performance Impact, Common breakdown</h3>
      <table class="simple-table">
        <thead class="simple-table-header">
          <tr>
            <th
              class="simple-table-header-color simple-table-header"
              style="width: 154.7447967529297px"
            >
              Thread
            </th>
            <th
              class="simple-table-header-color simple-table-header"
              style="width: 366.7447967529297px"
            >
              Job
            </th>
            <th
              class="simple-table-header-color simple-table-header"
              style="width: 86.73959350585938px"
            >
              Hardware
            </th>
            <th
              class="simple-table-header-color simple-table-header"
              style="width: 176.75px"
            >
              Blocking Effect
            </th>
          </tr>
        </thead>
        <tbody>
          <tr>
            <td class="" style="width: 154.7447967529297px">Main Thread</td>
            <td class="" style="width: 366.7447967529297px">
              JavaScript execution, DOM operations, Style calculation, Layout
            </td>
            <td class="" style="width: 86.73959350585938px">CPU</td>
            <td class="" style="width: 176.75px">
              Blocking freezes page interaction
            </td>
          </tr>
          <tr>
            <td class="" style="width: 154.7447967529297px">
              Compositor Thread
            </td>
            <td class="" style="width: 366.7447967529297px">
              Layer compositing, scrolling, transform/opacity animations
            </td>
            <td class="" style="width: 86.73959350585938px">CPU</td>
            <td class="" style="width: 176.75px">
              Runs independently, doesn&#x27;t block JavaScript
            </td>
          </tr>
          <tr>
            <td class="" style="width: 154.7447967529297px">Raster Thread</td>
            <td class="" style="width: 366.7447967529297px">
              Converts drawing commands to actual
              <mark class="highlight-red">pixels (note1)</mark>
            </td>
            <td class="" style="width: 86.73959350585938px">CPU/GPU</td>
            <td class="" style="width: 176.75px">
              Multi-core parallel, doesn&#x27;t block main flow
            </td>
          </tr>
          <tr>
            <td class="" style="width: 154.7447967529297px">GPU Process</td>
            <td class="" style="width: 366.7447967529297px">
              Hardware-accelerated rendering, WebGL, graphics compositing
            </td>
            <td class="" style="width: 86.73959350585938px">
              Integrated/Dedicated GPU
            </td>
            <td class="" style="width: 176.75px">
              Runs in background, performance depends on hardware
            </td>
          </tr>
          <tr>
            <td class="" style="width: 154.7447967529297px">Network Thread</td>
            <td class="" style="width: 366.7447967529297px">
              Sending and receiving HTTP requests
            </td>
            <td class="" style="width: 86.73959350585938px">
              Network<br />card1.
            </td>
            <td class="" style="width: 176.75px">
              Runs in background, doesn&#x27;t affect page rendering
            </td>
          </tr>
        </tbody>
      </table>
      <p class="">
        Real impact: This is why transform animations run smoother than width
        animations. Transform runs on the Compositor Thread and doesn&#x27;t
        block JavaScript. Width changes need Layout recalculation, which uses
        Main Thread resources and can cause page stuttering
      </p>
      <p class="">⚠️⚠️⚠️⚠️</p>
      <p class="">
        Only Main Thread work directly affects JavaScript execution and user
        interaction<br />responsiveness. Understanding which operations trigger
        which thread helps you write smoother interactions.
      </p>
      <p class=""></p>
      <p class="">
        <em><mark class="highlight-red">note1</mark></em
        ><em><mark class="highlight-gray">:</mark></em
        ><em><mark class="highlight-red"> </mark></em
        ><em
          ><mark class="highlight-gray"
            >A bitmap contains hundreds, thousands, or millions of pixels</mark
          ></em
        >
      </p>
      <p class=""></p>
      <p class=""></p>
      <p class=""></p>
      <h1 class="">CRP (Critical Render Path)</h1>
      <h3 class="">What is CRP?</h3>
      <p class="">
        CRP is the shortest path &quot;from hitting Enter to seeing the first
        screen.&quot; This decides
        <mark class="highlight-red">your website&#x27;s first impression</mark>
        and directly affects the FP (First Paint) metric in SEO.
      </p>
      <p class="">
        CRP only focuses on that &quot;from nothing to something&quot; moment
        during initial load, not the ongoing rendering process.
      </p>
      <h3 class="">CRP&#x27;s Six Key Stages</h3>
      <p class="">
        Browser rendering isn&#x27;t a single-threaded linear operation.
        Multiple specialized threads work together:
      </p>
      <ol type="1" class="numbered-list" start="1">
        <li>HTML builds DOM</li>
      </ol>
      <ol type="1" class="numbered-list" start="2">
        <li>
          CSS builds CSSOM -
          <mark class="highlight-red">parses style rules (note2)</mark>
        </li>
      </ol>
      <ol type="1" class="numbered-list" start="3">
        <li>
          Render Tree - combines DOM + CSSOM, determines which elements to show
        </li>
      </ol>
      <ol type="1" class="numbered-list" start="4">
        <li>
          Layout/Reflow - calculates exact positions and sizes of elements
        </li>
      </ol>
      <ol type="1" class="numbered-list" start="5">
        <li>
          Paint - converts elements to actual pixels (Raster Thread handles
          this)
        </li>
      </ol>
      <ol type="1" class="numbered-list" start="6">
        <li>
          Composite - combines all layers into final screen output (Compositor
          Thread handles this)
        </li>
      </ol>
      <p class="">
        During the Render Tree stage, users still can&#x27;t see anything. This
        stage only confirms that DOM is ready, CSS is parsed, and element
        relationships are established (including which elements to hide like
        <mark class="highlight-red">display: none</mark>), but there&#x27;s no
        actual position data, pixel data, or final output yet.
      </p>
      <p class="">
        <em><mark class="highlight-red">note2</mark></em
        ><em
          ><mark class="highlight-gray"
            >: useragent stylesheet → external → internal (style tag) →
            inline(style=””)</mark
          ></em
        >
      </p>
      <h3 class="">CRP vs General Rendering Pipeline Differences</h3>
      <p class="">
        While the steps look similar, their focus is completely different:
      </p>
      <ul class="bulleted-list">
        <li style="list-style-type: disc">
          CRP (one-time):
          <p class="">
            HTML → DOM → CSSOM → Layout → Paint → Composite → First visible
          </p>
          <p class="">
            Focuses on necessary steps before &quot;first visible,&quot; affects
            FP and FCP metrics.
          </p>
        </li>
      </ul>
      <ul class="bulleted-list">
        <li style="list-style-type: disc">
          Rendering Pipeline (continuous loop):
          <p class="">
            rAF → Style → Layout → Paint → Composite → Next frame (16ms)
          </p>
          <p class="">
            Complete rendering flow for each frame, continuously maintains
            screen updates.
          </p>
        </li>
      </ul>
      <h3 class="">Different Focus Areas for CRP Optimization</h3>
      <p class="">
        CRP optimization directly improves your FCP and FP scores, while LCP
        optimization has different priorities. Understanding both differences
        helps create the right optimization strategy.
      </p>
      <p class="">
        FCP focuses on &quot;first content&quot; appearance speed, emphasizing
        server response and critical resource loading:
      </p>
      <ul class="bulleted-list">
        <li style="list-style-type: disc">TTFB: server performance</li>
      </ul>
      <ul class="bulleted-list">
        <li style="list-style-type: disc">
          async/defer script strategies: avoid synchronous scripts blocking DOM
          parsing and first render
        </li>
      </ul>
      <ul class="bulleted-list">
        <li style="list-style-type: disc">
          Critical CSS inline: ensure above-the-fold styles are immediately
          available without waiting for external CSS files
        </li>
      </ul>
      <ul class="bulleted-list">
        <li style="list-style-type: disc">
          Remove render-blocking resources: delay non-critical CSS and
          JavaScript loading
        </li>
      </ul>
      <p class=""></p>
      <p class="">
        LCP focuses on &quot;largest content element&quot; loading completion,
        leaning toward client-side performance and resource priority:
      </p>
      <ul class="bulleted-list">
        <li style="list-style-type: disc">
          Code performance:
          <mark class="highlight-red">reduce Main Thread blocking</mark>, giving
          browser more time to process large content
        </li>
      </ul>
      <ul class="bulleted-list">
        <li style="list-style-type: disc">
          Caching mechanisms: ensure important resources load quickly
        </li>
      </ul>
      <ul class="bulleted-list">
        <li style="list-style-type: disc">
          preload/prefetch strategies: pre-load LCP candidate elements (usually
          large images or important content)
        </li>
      </ul>
      <ul class="bulleted-list">
        <li style="list-style-type: disc">
          lazy loading implementation: delay non-critical image loading to avoid
          bandwidth competition
        </li>
      </ul>
      <p class="">
        Implementation: For images,
        <mark class="highlight-red">loading=eager</mark> ensures critical
        resources come first, improving FCP. Others use
        <mark class="highlight-red">loading=lazy</mark> to avoid bandwidth
        competition. <mark class="highlight-red">fetchpriority=high</mark> lets
        hero images get priority, optimizing LCP metrics. But usually image
        loading strategies are more related to LCP.
      </p>
      <p class="">
        Technical details: <mark class="highlight-red">loading=lazy</mark> uses
        <mark class="highlight-red">IntersectionObserver</mark> to determine
        when to load, while
        <mark class="highlight-red">fetchpriority</mark> affects request queue
        priority ordering.
      </p>
      <p class=""></p>
      <h3 class="">Web Performance Metrics Sources</h3>
      <p class="">
        These SEO metrics are direct products of browser-level Paint and
        Composite events, but their generation and monitoring mechanisms are
        more complex than expected.
      </p>
      <table class="simple-table">
        <thead class="simple-table-header">
          <tr>
            <th class="simple-table-header-color simple-table-header">
              Metric
            </th>
            <th class="simple-table-header-color simple-table-header">
              Definition
            </th>
            <th class="simple-table-header-color simple-table-header">
              Trigger Timing
            </th>
            <th class="simple-table-header-color simple-table-header">
              Approximate
            </th>
          </tr>
        </thead>
        <tbody>
          <tr>
            <td class="">First Paint (FP)</td>
            <td class="">First pixel drawn</td>
            <td class="">Paint stage<br />begins</td>
            <td class="">200~500ms</td>
          </tr>
          <tr>
            <td class="">First Contentful Paint (FCP)</td>
            <td class="">First content element drawn</td>
            <td class="">Meaningful<br />content Paint</td>
            <td class="">300~600ms</td>
          </tr>
          <tr>
            <td class="">Largest Contentful Paint (LCP)</td>
            <td class="">Largest content element completes rendering</td>
            <td class="">Main content<br />Paint</td>
            <td class="">1000~2500ms</td>
          </tr>
          <tr>
            <td class="">DOMContentLoaded</td>
            <td class="">DOM structure complete</td>
            <td class="">DOM parsing<br />finished</td>
            <td class="">100~300ms</td>
          </tr>
          <tr>
            <td class="">load</td>
            <td class="">All resources loaded</td>
            <td class="">All resources<br />loaded</td>
            <td class="">1200~3000ms</td>
          </tr>
        </tbody>
      </table>
      <p class="">Complete flow of metric generation:</p>
      <p class="">
        These metric triggers are indeed related to Main Thread, but the entire
        mechanism spans multiple threads:
      </p>
      <p class="">FP (First Paint) generation process:</p>
      <ol type="1" class="numbered-list" start="1">
        <li>Main Thread completes first Paint command</li>
      </ol>
      <ol type="1" class="numbered-list" start="2">
        <li>Raster Thread executes actual drawing, generates pixel data</li>
      </ol>
      <ol type="1" class="numbered-list" start="3">
        <li>Compositor Thread composites pixels to screen</li>
      </ol>
      <ol type="1" class="numbered-list" start="4">
        <li>
          Browser internally detects &quot;first pixel&quot; appearance,
          triggers FP event
        </li>
      </ol>
      <p class=""></p>
      <p class="">
        FCP and LCP detection mechanism: Same flow through Main Thread → Raster
        Thread → Compositor Thread, but browser internal algorithms further
        analyze which are &quot;meaningful content&quot; or &quot;largest
        content elements,&quot; then trigger corresponding events.
      </p>
      <p class=""></p>
      <h3 class="">How to Monitor These SEO Metrics</h3>
      <p class="">
        While metric generation spans multiple Threads, they&#x27;re all exposed
        to developers through Performance API in Main Thread:
      </p>
      <p class="">
        <mark class="highlight-red"
          ><strong
            >Browser automatically detects at low level, JS passively
            receives</strong
          ></mark
        >
      </p>
      <script
        src="https://cdnjs.cloudflare.com/ajax/libs/prism/1.29.0/prism.min.js"
        integrity="sha512-7Z9J3l1+EYfeaPKcGXu3MS/7T+w19WtKQY/n+xzmw4hZhJ9tyYmcUS+4QqAlzhicE5LAfMQSF3iFTK9bQdTxXg=="
        crossorigin="anonymous"
        referrerpolicy="no-referrer"
      ></script>
      <link
        rel="stylesheet"
        href="https://cdnjs.cloudflare.com/ajax/libs/prism/1.29.0/themes/prism.min.css"
        integrity="sha512-tN7Ec6zAFaVSG3TpNAKtk4DOHNpSwKHxxrsiw4GHKESGPs5njn/0sMCUMl2svV4wo4BK/rCP7juYz+zx+l6oeQ=="
        crossorigin="anonymous"
        referrerpolicy="no-referrer"
      />
      <pre
        class="code code-wrap"
      ><code class="language-JavaScript" style="white-space:pre-wrap;word-break:break-all">const observer = new PerformanceObserver((list) =&gt; {
  for (const entry of list.getEntries()) {
    if (entry.name === &#x27;first-contentful-paint&#x27;) {
      console.log(&#x27;FCP:&#x27;, entry.startTime);
    }
    if (entry.entryType === &#x27;largest-contentful-paint&#x27;) {
      console.log(&#x27;LCP:&#x27;, entry.startTime);
    }
  }
});

observer.observe({entryTypes: [&#x27;paint&#x27;, &#x27;largest-contentful-paint&#x27;]});
// paint: fp, fcp
// largest-contentful-paint: only for LCP</code></pre>
      <p class="">
        These metrics don&#x27;t directly bind JavaScript callbacks, but are
        events from browser-level drawing stages, accessible through Chrome
        Performance Timeline or Paint Timing API.
      </p>
      <p class="">
        <mark class="highlight-red"
          >&quot;Don&#x27;t directly bind JavaScript callbacks&quot; means you
          can&#x27;t use addEventListener.</mark
        >
      </p>
      <p class="">
        <a href="https://web.dev/articles/fcp">https://web.dev/articles/fcp</a>
      </p>
      <p class="">
        <a href="https://web.dev/articles/lcp">https://web.dev/articles/lcp</a>
      </p>
      <h3 class="">CSS Properties&#x27; Impact on Rendering Stages</h3>
      <p class=""></p>
      <p class="">
        Browser rendering follows waterfall principle: triggering Layout will
        cascade through all subsequent stages.
      </p>
      <table class="simple-table">
        <thead class="simple-table-header">
          <tr>
            <th
              class="simple-table-header-color simple-table-header"
              style="width: 121.65972900390625px"
            >
              Trigger Stage
            </th>
            <th
              class="simple-table-header-color simple-table-header"
              style="width: 452.6562805175781px"
            >
              CSS Properties
            </th>
            <th
              class="simple-table-header-color simple-table-header"
              style="width: 87.65972900390625px"
            >
              Performance
            </th>
          </tr>
        </thead>
        <tbody>
          <tr>
            <td class="" style="width: 121.65972900390625px">Layout</td>
            <td class="" style="width: 452.6562805175781px">
              width, margin, padding, top, font-size, border-width,
              position:fixed
            </td>
            <td class="" style="width: 87.65972900390625px">badly</td>
          </tr>
          <tr>
            <td class="" style="width: 121.65972900390625px">Paint</td>
            <td class="" style="width: 452.6562805175781px">
              color, background, border-color, box-shadow, outline
            </td>
            <td class="" style="width: 87.65972900390625px">so-so</td>
          </tr>
          <tr>
            <td class="" style="width: 121.65972900390625px">Composite</td>
            <td class="" style="width: 452.6562805175781px">
              transform, opacity, filter, will-change
            </td>
            <td class="" style="width: 87.65972900390625px">nice</td>
          </tr>
        </tbody>
      </table>
      <p class="">
        Animations should prioritize Composite properties to avoid unnecessary
        recalculations:
      </p>
      <table class="simple-table">
        <thead class="simple-table-header">
          <tr>
            <th class="simple-table-header-color simple-table-header">
              Desired Effect
            </th>
            <th class="simple-table-header-color simple-table-header">
              ❌ Avoid Using
            </th>
            <th class="simple-table-header-color simple-table-header">
              ✅ Recommended
            </th>
          </tr>
        </thead>
        <tbody>
          <tr>
            <td class="">Move element</td>
            <td class="">left, top, margin</td>
            <td class="">transform: translateX/Y()</td>
          </tr>
          <tr>
            <td class="">Scale element</td>
            <td class="">width, height</td>
            <td class="">transform: scale()</td>
          </tr>
          <tr>
            <td class="">Hide element</td>
            <td class="">display: none/block</td>
            <td class="">opacity: 0/1</td>
          </tr>
        </tbody>
      </table>
      <p class=""></p>
      <p class=""></p>
      <h1 class="">
        Layer Management and Animation Optimization: Running CSS on the Right
        Thread
      </h1>
      <p class="">
        Earlier we learned about the
        <strong>browser&#x27;s Compositor Thread</strong> and
        <strong>CRP rendering flow</strong>. Now let&#x27;s see how to use this
        knowledge for performance optimization.
      </p>
      <h3 class="">Browser Layer Creation Strategy</h3>
      <p class="">
        Browsers don&#x27;t create independent composite layers for every
        element - that would use too much memory. Instead, they decide whether
        to promote elements to independent layers based on specific conditions.
        When elements meet these conditions,
        <mark class="highlight-red"
          >the browser lets the Compositor Thread handle them, avoiding Main
          Thread resource usage.</mark
        >
      </p>
      <p class="">
        Transform property rules deserve special attention. 3D transforms (like
        <mark class="highlight-red">translateZ(0)</mark> or
        <mark class="highlight-red">translate3d()</mark>) immediately create
        layers, but 2D transforms (like
        <mark class="highlight-red">translateX(50px)</mark>) usually don&#x27;t
        in static states.
      </p>
      <p class="">
        However, adding <mark class="highlight-red">will-change</mark>:
        transform forces the browser to create a layer. During animation
        execution, even 2D transforms temporarily create layers.
      </p>
      <p class="">
        Another common layer trigger is position: fixed. Fixed position elements
        need independent calculations relative to the viewport, so browsers
        promote them to independent layers to improve scrolling performance.
      </p>
      <p class=""></p>
      <h3 class="">Three Levels of Animation Performance</h3>
      <p class="">
        Not all animations can enjoy Compositor Thread optimization. The key is
        which rendering stage the animation properties trigger.
      </p>
      <ul class="bulleted-list">
        <li style="list-style-type: disc">
          Composite-level animations
          <mark class="highlight-red">(transform, opacity, filter)</mark> are
          the best performance choice. These animations create layers and run on
          the Compositor Thread, completely avoiding blocking JavaScript or user
          interactions.
        </li>
      </ul>
      <p class="">⚠️⚠️⚠️</p>
      <ul class="bulleted-list">
        <li style="list-style-type: disc">
          Paint-level animations
          <mark class="highlight-red">(color, background)</mark> don&#x27;t
          create layers but only need pixel repainting without affecting element
          layout. Medium performance.
        </li>
      </ul>
      <p class="">⚠️⚠️⚠️</p>
      <ul class="bulleted-list">
        <li style="list-style-type: disc">
          Layout-level animations
          <mark class="highlight-red">(width, height, margin)</mark> are the
          most performance-<mark class="highlight-red"
            ><strong>expensive</strong></mark
          >
          choice. These animations trigger complete reflow processes, using lots
          of Main Thread resources and potentially causing page stuttering.
        </li>
      </ul>
      <h3 class="">Does Hover Create New Layers?</h3>
      <p class="">
        A common misconception is that hover states automatically create layers.
        Actually, hover itself has no relationship with layers, unless the hover
        effect triggers animations that create layers.
      </p>
      <p class="">
        Best practice strategy is prioritizing
        <mark class="highlight-red"
          >using transform and opacity for animations</mark
        >. For elements expected to animate frequently, use
        <mark class="highlight-red">will-change: transform</mark> to pre-create
        layers, but remember to remove this property after animation ends to
        free resources. More importantly, avoid using Layout-level properties
        for animations - this puts unnecessary burden on the Main Thread and
        affects overall user experience.
      </p>
      <p class=""></p>
      <p class=""></p>
      <p class=""></p>
      <p class=""></p>
      <p class=""></p>
      <h1 class="">requestAnimationFrame&#x27;s Role and Timing</h1>
      <p class="">
        rAF isn&#x27;t just an animation API - it&#x27;s a key player in the
        browser rendering process.<br />Understanding rAF&#x27;s execution
        timing, relationship with Event Loop, and behavior in modern frameworks
        is crucial for writing smooth animations and avoiding performance traps.
      </p>
      <p class="">This section includes:</p>
      <ul class="bulleted-list">
        <li style="list-style-type: disc">
          <strong>rAF&#x27;s basic mechanism</strong>: Why it&#x27;s not a
          timer, but a reservation system
        </li>
      </ul>
      <ul class="bulleted-list">
        <li style="list-style-type: disc">
          Position in rendering flow: rAF&#x27;s unique spot between Event Loop
          and Rendering Pipeline
        </li>
      </ul>
      <ul class="bulleted-list">
        <li style="list-style-type: disc">
          <strong>Frame dropping analysis</strong>: What happens when JavaScript
          execution times out, how to detect and avoid it
        </li>
      </ul>
      <ul class="bulleted-list">
        <li style="list-style-type: disc">
          <strong>Cooperation with Event Loop</strong>: How Microtasks affect
          animation smoothness, and execution order of<br />different async
          tasks
        </li>
      </ul>
      <ul class="bulleted-list">
        <li style="list-style-type: disc">
          <strong>Real applications in React framework</strong>: Timing
          relationship between lifecycle and rAF, and how batch updates affect
          animation timing
        </li>
      </ul>
      <p class=""></p>
      <h3 class="">Browser&#x27;s 16ms Metronome</h3>
      <p class="">
        Modern browsers work like precise metronomes, but the beat speed adjusts
        to your screen refresh rate. Traditional 60Hz screens update every 16.67
        milliseconds, but if your computer is 120Hz, then it&#x27;s every 8.33
        milliseconds. Browsers automatically detect screen refresh rate and
        adjust rAF execution frequency.
      </p>
      <p class="">
        <mark class="highlight-red">rAF is actually a reservation system</mark>,
        letting you book function execution before the next render frame. When
        you call it, the browser queues your function for the next frame&#x27;s
        execution list, but only executes once then stops. For continuous
        animation, you must re-book each time:
      </p>
      <script
        src="https://cdnjs.cloudflare.com/ajax/libs/prism/1.29.0/prism.min.js"
        integrity="sha512-7Z9J3l1+EYfeaPKcGXu3MS/7T+w19WtKQY/n+xzmw4hZhJ9tyYmcUS+4QqAlzhicE5LAfMQSF3iFTK9bQdTxXg=="
        crossorigin="anonymous"
        referrerpolicy="no-referrer"
      ></script>
      <link
        rel="stylesheet"
        href="https://cdnjs.cloudflare.com/ajax/libs/prism/1.29.0/themes/prism.min.css"
        integrity="sha512-tN7Ec6zAFaVSG3TpNAKtk4DOHNpSwKHxxrsiw4GHKESGPs5njn/0sMCUMl2svV4wo4BK/rCP7juYz+zx+l6oeQ=="
        crossorigin="anonymous"
        referrerpolicy="no-referrer"
      />
      <pre
        class="code code-wrap"
      ><code class="language-JavaScript" style="white-space:pre-wrap;word-break:break-all">function animate() {
  // Do some animation updates
  requestAnimationFrame(animate); // Re-book next time
}
requestAnimationFrame(animate); // Start animation loop</code></pre>
      <p class="">
        Multiple reservations within the same frame are possible. If you
        register multiple rAF callbacks in the previous frame, they&#x27;ll
        execute in registration order within the same time window:
      </p>
      <script
        src="https://cdnjs.cloudflare.com/ajax/libs/prism/1.29.0/prism.min.js"
        integrity="sha512-7Z9J3l1+EYfeaPKcGXu3MS/7T+w19WtKQY/n+xzmw4hZhJ9tyYmcUS+4QqAlzhicE5LAfMQSF3iFTK9bQdTxXg=="
        crossorigin="anonymous"
        referrerpolicy="no-referrer"
      ></script>
      <link
        rel="stylesheet"
        href="https://cdnjs.cloudflare.com/ajax/libs/prism/1.29.0/themes/prism.min.css"
        integrity="sha512-tN7Ec6zAFaVSG3TpNAKtk4DOHNpSwKHxxrsiw4GHKESGPs5njn/0sMCUMl2svV4wo4BK/rCP7juYz+zx+l6oeQ=="
        crossorigin="anonymous"
        referrerpolicy="no-referrer"
      />
      <pre
        class="code code-wrap"
      ><code class="language-JavaScript" style="white-space:pre-wrap;word-break:break-all">requestAnimationFrame((ts1) =&gt; {
  const frame1 = Math.floor(ts1 / 8.33); // 120Hz frame calculation
  console.log(`First rAF: ${ts1.toFixed(1)}ms (Frame ${frame1})`);

  requestAnimationFrame((ts2) =&gt; {
    const frame2 = Math.floor(ts2 / 8.33);
    console.log(`Callback 1: ${ts2.toFixed(1)}ms (Frame ${frame2})`);
  });

  requestAnimationFrame((ts3) =&gt; {
    const frame3 = Math.floor(ts3 / 8.33);
    console.log(`Callback 2: ${ts3.toFixed(1)}ms (Frame ${frame3})`);
  });
});</code></pre>
      <p class=""></p>
      <h3 class="">rAF&#x27;s True Role: Final Adjustments Before Rendering</h3>
      <blockquote class="">
        <mark class="highlight-red"
          >1 frame (16ms): event loop → raf → [Style
          Calc,Layout,Paint,Composite]</mark
        >
      </blockquote>
      <p class="">
        rAF is the browser&#x27;s &quot;final adjustment mechanism before
        rendering.&quot; In each frame&#x27;s rendering flow,
        <mark class="highlight-red"
          >it gives you one last chance to modify DOM before Style Calculation,
          Layout, Paint, and Composite begin.</mark
        >
      </p>
      <p class="">Complete frame execution order: Frame Timeline (16ms)</p>
      <ol type="1" class="numbered-list" start="1">
        <li>Call stack completes (synchronous JS code)</li>
      </ol>
      <ol type="1" class="numbered-list" start="2">
        <li>Clear all Microtasks (Promise.then, queueMicrotask)</li>
      </ol>
      <ol type="1" class="numbered-list" start="3">
        <li>Execute Macrotask (setTimeout, DOM events)</li>
      </ol>
      <ol type="1" class="numbered-list" start="4">
        <li>rAF callbacks (if rendering needed)</li>
      </ol>
      <ol type="1" class="numbered-list" start="5">
        <li>Rendering flow (Style → Layout → Paint → Composite)</li>
      </ol>
      <script
        src="https://cdnjs.cloudflare.com/ajax/libs/prism/1.29.0/prism.min.js"
        integrity="sha512-7Z9J3l1+EYfeaPKcGXu3MS/7T+w19WtKQY/n+xzmw4hZhJ9tyYmcUS+4QqAlzhicE5LAfMQSF3iFTK9bQdTxXg=="
        crossorigin="anonymous"
        referrerpolicy="no-referrer"
      ></script>
      <link
        rel="stylesheet"
        href="https://cdnjs.cloudflare.com/ajax/libs/prism/1.29.0/themes/prism.min.css"
        integrity="sha512-tN7Ec6zAFaVSG3TpNAKtk4DOHNpSwKHxxrsiw4GHKESGPs5njn/0sMCUMl2svV4wo4BK/rCP7juYz+zx+l6oeQ=="
        crossorigin="anonymous"
        referrerpolicy="no-referrer"
      />
      <pre
        class="code code-wrap"
      ><code class="language-Plain Text" style="white-space:pre-wrap;word-break:break-all">Frame Timeline (16ms)
├─ 0ms ─────────────── 16ms ─┤

1. Call stack completes
   └── Synchronous JS code

2. Clear all Microtasks
   └── Promise.then, queueMicrotask...

3. Execute Macrotask (until time budget runs out)
   └── setTimeout, DOM events...

4. rAF callbacks (if rendering needed)
   └── requestAnimationFrame

5. Rendering flow (if DOM changes exist)
   └── Style → Layout → Paint → Composite

└─ Next frame begins ─────────────────┤</code></pre>
      <p class=""></p>
      <p class="">
        Imagine this: JavaScript changes element to
        <mark class="highlight-red">red</mark>, but during rAF callback changes
        it to <mark class="highlight-blue">blue</mark>. The browser will merge
        these changes due to batch processing, finally displaying
        <mark class="highlight-blue">blue</mark>.
      </p>
      <p class="">
        <strong
          >This is rAF&#x27;s purpose: giving you one last adjustment chance
          before actual rendering.</strong
        >
      </p>
      <p class=""></p>
      <p class=""></p>
      <p class=""></p>
      <h3 class="">
        rAF&#x27;s Position: Bridge Between Event Loop and Rendering
      </h3>
      <p class="">
        rAF is hard to categorize. Say it&#x27;s part of Event Loop, but it
        doesn&#x27;t have fixed execution timing like setTimeout or Promise. Say
        it&#x27;s part of Rendering, but it doesn&#x27;t do any actual drawing
        work.
      </p>
      <p class=""></p>
      <p class="">The actual situation is this:</p>
      <p class="">
        After each Event Loop round completes, the browser checks if screen
        updates are needed. If needed, it executes all queued rAF callbacks,
        then starts the real rendering work (Style, Layout, Paint stuff). So rAF
        is like the &quot;final ultimatum&quot; before rendering:
      </p>
      <blockquote class="">
        want to change DOM? Now&#x27;s your last chance, miss this village and
        there&#x27;s no next shop.
      </blockquote>
      <p class="">
        The browser is about to start calculating styles, layout, and drawing.
        This is why changing styles in rAF runs smoother than in setTimeout.
        setTimeout might execute at any time during the render cycle, but rAF
        guarantees execution before rendering, avoiding unnecessary repaints.
      </p>
      <p class="">
        <strong
          >Simply put: rAF is the moment when the browser says &quot;I&#x27;m
          about to start drawing, do you have anything else to change?</strong
        >
      </p>
      <p class=""></p>
      <h3 class="">Frame Dropping: When Time Budget Goes Over</h3>
      <p class="">
        Frame dropping means the browser can&#x27;t complete a frame update
        within expected time, causing screen stuttering. For 60Hz screens,
        updates should happen every 16ms; for 120Hz, every 8.33ms. When
        JavaScript execution time exceeds this budget, it causes frame skipping.
      </p>
      <p class=""></p>
      <p class="">
        When JavaScript execution takes too long, like some code blocking for
        40ms, it disrupts this rhythm. The rAF callback that should execute at
        16ms gets delayed to 50ms, causing frame skipping. Users expect
        animation updates every 16ms, but now it becomes updates every 33ms,
        50ms, or even longer intervals, making the screen appear
        &quot;stuttery.&quot;
      </p>
      <p class=""></p>
      <p class=""><strong>Frame dropping test:</strong></p>
      <p class="">
        This frame dropping phenomenon is obvious in Chrome, more obvious when
        adjusting &quot;4x slowdown&quot; in DevTools Performance. Firefox
        performs relatively less obviously.
      </p>
      <script
        src="https://cdnjs.cloudflare.com/ajax/libs/prism/1.29.0/prism.min.js"
        integrity="sha512-7Z9J3l1+EYfeaPKcGXu3MS/7T+w19WtKQY/n+xzmw4hZhJ9tyYmcUS+4QqAlzhicE5LAfMQSF3iFTK9bQdTxXg=="
        crossorigin="anonymous"
        referrerpolicy="no-referrer"
      ></script>
      <link
        rel="stylesheet"
        href="https://cdnjs.cloudflare.com/ajax/libs/prism/1.29.0/themes/prism.min.css"
        integrity="sha512-tN7Ec6zAFaVSG3TpNAKtk4DOHNpSwKHxxrsiw4GHKESGPs5njn/0sMCUMl2svV4wo4BK/rCP7juYz+zx+l6oeQ=="
        crossorigin="anonymous"
        referrerpolicy="no-referrer"
      />
      <pre
        class="code code-wrap"
      ><code class="language-JavaScript" style="white-space:pre-wrap;word-break:break-all">const globalStart = performance.now();

function busyLoop(ms) {
  const end = performance.now() + ms;
  while (performance.now() &lt; end) {}
}

function animate(now) {
  const currentTime = performance.now() - globalStart;
  console.log(`rAF execution: ${currentTime.toFixed(1)}ms`);

  document.body.style.transform = `translateX(${Math.sin(now / 100) * 100}px)`;
  requestAnimationFrame(animate);
}

requestAnimationFrame(animate);

// Simulate JS blocking
setInterval(() =&gt; {
  const beforeBlock = performance.now() - globalStart;
  console.log(`setInterval starts blocking: ${beforeBlock.toFixed(1)}ms`);

  busyLoop(100); // Block for 100ms

  const afterBlock = performance.now() - globalStart;
  console.log(`setInterval blocking ends: ${afterBlock.toFixed(1)}ms`);
}, 1000);</code></pre>
      <p class=""></p>
      <p class=""></p>
      <p class=""></p>
      <h3 class="">Cooperation with Event Loop</h3>
      <p class="">
        In browsers, Micro and Macro tasks don&#x27;t have priority concepts,
        only execution timing rules.
      </p>
      <p class="">
        Microtasks might delay rAF execution. If Promise.then contains
        time-consuming operations, it directly affects animation smoothness,
        because Microtasks must all complete before rAF.
      </p>
      <script
        src="https://cdnjs.cloudflare.com/ajax/libs/prism/1.29.0/prism.min.js"
        integrity="sha512-7Z9J3l1+EYfeaPKcGXu3MS/7T+w19WtKQY/n+xzmw4hZhJ9tyYmcUS+4QqAlzhicE5LAfMQSF3iFTK9bQdTxXg=="
        crossorigin="anonymous"
        referrerpolicy="no-referrer"
      ></script>
      <link
        rel="stylesheet"
        href="https://cdnjs.cloudflare.com/ajax/libs/prism/1.29.0/themes/prism.min.css"
        integrity="sha512-tN7Ec6zAFaVSG3TpNAKtk4DOHNpSwKHxxrsiw4GHKESGPs5njn/0sMCUMl2svV4wo4BK/rCP7juYz+zx+l6oeQ=="
        crossorigin="anonymous"
        referrerpolicy="no-referrer"
      />
      <pre
        class="code code-wrap"
      ><code class="language-JavaScript" style="white-space:pre-wrap;word-break:break-all">// Show Microtask blocking rAF problem
Promise.resolve().then(() =&gt; {
  // Simulate time-consuming Microtask
  const start = performance.now();
  while (performance.now() - start &lt; 5000) {} // Block for 5000ms
  console.log(&#x27;Microtask complete, but blocked animation&#x27;);
});

requestAnimationFrame(() =&gt; {
  console.log(&#x27;rAF delayed execution, animation might stutter&#x27;);
});</code></pre>
      <p class="">Complex execution order example:</p>
      <p class="">
        The setTimeout here, if delay is greater than 0ms, has some chance of
        executing later than rAF 2, because setTimeout(fn, 0) actual minimum
        delay is 4ms (HTML5 standard), plus Macrotask execution timing, might
        get pushed to next frame.
      </p>
      <script
        src="https://cdnjs.cloudflare.com/ajax/libs/prism/1.29.0/prism.min.js"
        integrity="sha512-7Z9J3l1+EYfeaPKcGXu3MS/7T+w19WtKQY/n+xzmw4hZhJ9tyYmcUS+4QqAlzhicE5LAfMQSF3iFTK9bQdTxXg=="
        crossorigin="anonymous"
        referrerpolicy="no-referrer"
      ></script>
      <link
        rel="stylesheet"
        href="https://cdnjs.cloudflare.com/ajax/libs/prism/1.29.0/themes/prism.min.css"
        integrity="sha512-tN7Ec6zAFaVSG3TpNAKtk4DOHNpSwKHxxrsiw4GHKESGPs5njn/0sMCUMl2svV4wo4BK/rCP7juYz+zx+l6oeQ=="
        crossorigin="anonymous"
        referrerpolicy="no-referrer"
      />
      <pre
        class="code code-wrap"
      ><code class="language-JavaScript" style="white-space:pre-wrap;word-break:break-all">  console.log(&#x27;script start&#x27;);

  setTimeout(() =&gt; console.log(&#x27;setTimeout&#x27;), 0);
  Promise.resolve().then(() =&gt; console.log(&#x27;microtask&#x27;));

  requestAnimationFrame(() =&gt; {
    console.log(&#x27;rAF 1&#x27;);
    requestAnimationFrame(() =&gt; {
      console.log(&#x27;rAF 2&#x27;);
    });
  });

  console.log(&#x27;script end&#x27;);

  /*
  Output order:
  script start
  script end
  microtask
  rAF 1
  setTimeout
  rAF 2
  */</code></pre>
      <h3 class="">rAF Applications in React Framework</h3>
      <p class="">
        Through testing we can observe that React&#x27;s update flow has clear
        timing relationships with rAF callbacks.
      </p>
      <p class=""></p>
      <ul class="toggle">
        <li>
          <details open="">
            <summary>Registering rAF in event handlers</summary>
            <script
              src="https://cdnjs.cloudflare.com/ajax/libs/prism/1.29.0/prism.min.js"
              integrity="sha512-7Z9J3l1+EYfeaPKcGXu3MS/7T+w19WtKQY/n+xzmw4hZhJ9tyYmcUS+4QqAlzhicE5LAfMQSF3iFTK9bQdTxXg=="
              crossorigin="anonymous"
              referrerpolicy="no-referrer"
            ></script>
            <link
              rel="stylesheet"
              href="https://cdnjs.cloudflare.com/ajax/libs/prism/1.29.0/themes/prism.min.css"
              integrity="sha512-tN7Ec6zAFaVSG3TpNAKtk4DOHNpSwKHxxrsiw4GHKESGPs5njn/0sMCUMl2svV4wo4BK/rCP7juYz+zx+l6oeQ=="
              crossorigin="anonymous"
              referrerpolicy="no-referrer"
            />
            <pre
              class="code code-wrap"
            ><code class="language-JavaScript" style="white-space:pre-wrap;word-break:break-all">const handleClick = () =&gt; {
  const clickTime = performance.now();
  console.log(`1. Click event: ${clickTime.toFixed(1)}ms`);

  requestAnimationFrame((rafTime) =&gt; {
    console.log(`4. rAF callback: ${rafTime.toFixed(1)}ms`);
  });

  setCount(c =&gt; c + 1);
  console.log(`2. setState complete: ${performance.now().toFixed(1)}ms`);
};

console.log(`3. render: ${performance.now().toFixed(1)}ms`);

useLayoutEffect(() =&gt; {
  console.log(`5. useLayoutEffect: ${performance.now().toFixed(1)}ms`);
}, [count]);

useEffect(() =&gt; {
  console.log(`6. useEffect: ${performance.now().toFixed(1)}ms`);
}, [count]);

/*
Typical output:
1. Click event: 1758.0ms
2. setState complete: 1759.0ms
3. render: 1759.0ms
5. useLayoutEffect: 1759.0ms
6. useEffect: 1759.0ms
4. rAF callback: 1769.7ms (delayed about 10ms)
*/</code></pre>
          </details>
        </li>
      </ul>
      <p class="">
        Observation results: setState/render/useLayoutEffect/useEffect execute
        almost simultaneously, rAF callback follows shortly after
      </p>
      <p class=""></p>
      <p class=""></p>
      <ul class="toggle">
        <li>
          <details open="">
            <summary>Registering rAF in React lifecycle</summary>
            <script
              src="https://cdnjs.cloudflare.com/ajax/libs/prism/1.29.0/prism.min.js"
              integrity="sha512-7Z9J3l1+EYfeaPKcGXu3MS/7T+w19WtKQY/n+xzmw4hZhJ9tyYmcUS+4QqAlzhicE5LAfMQSF3iFTK9bQdTxXg=="
              crossorigin="anonymous"
              referrerpolicy="no-referrer"
            ></script>
            <link
              rel="stylesheet"
              href="https://cdnjs.cloudflare.com/ajax/libs/prism/1.29.0/themes/prism.min.css"
              integrity="sha512-tN7Ec6zAFaVSG3TpNAKtk4DOHNpSwKHxxrsiw4GHKESGPs5njn/0sMCUMl2svV4wo4BK/rCP7juYz+zx+l6oeQ=="
              crossorigin="anonymous"
              referrerpolicy="no-referrer"
            />
            <pre
              class="code code-wrap"
            ><code class="language-JavaScript" style="white-space:pre-wrap;word-break:break-all">useLayoutEffect(() =&gt; {
  const startTime = performance.now();
  const startFrame = Math.floor(startTime / 8.33); // 120Hz calculation

  console.log(
    `%cuseLayoutEffect%c: ${startTime.toFixed(1)}ms (Frame ${startFrame})`,
    &#x27;background: #2196F3; color: white; padding: 2px 4px;&#x27;, &#x27;&#x27;
  );

  requestAnimationFrame(() =&gt; {
    const rafTime = performance.now();
    const rafFrame = Math.floor(rafTime / 8.33);
    const frameDiff = rafFrame - startFrame;
    const timeDiff = (rafTime - startTime).toFixed(1);

    // Decide background color based on delayed frames
    const bgColor = frameDiff &gt;= 2 ? &#x27;#F44336&#x27; : &#x27;#4CAF50&#x27;;

    console.log(
      `%crAF Callback%c: ${rafTime.toFixed(1)}ms (Frame ${rafFrame}) | Delayed ${frameDiff}
frames | ${timeDiff}ms`,
      `background: ${bgColor}; color: white; padding: 2px 4px; font-weight: bold;`, &#x27;&#x27;
    );
  });
}, [count]);</code></pre>
          </details>
        </li>
      </ul>
      <p class="">
        When registering rAF in useLayoutEffect, the delay phenomenon becomes
        more obvious:
      </p>
      <p class=""></p>
      <p class="">
        React completes all updates within logical frames (render →
        useLayoutEffect → useEffect) almost simultaneously (&lt; 1ms
        difference).
      </p>
      <p class="">
        Key finding: rAF callbacks registered in React lifecycle methods delay
        more than those in event handlers. Possible reasons:
      </p>
      <ol type="1" class="numbered-list" start="1">
        <li>
          Event handlers register rAF directly, catching the current
          frame&#x27;s time window
        </li>
      </ol>
      <ol type="1" class="numbered-list" start="2">
        <li>
          Lifecycle methods go through React Scheduler, causing later
          registration timing and missing the current frame<br /><mark
            class="highlight-blue"
            ><strong
              >(But I&#x27;m just speculating, going deeper is beyond my current
              ability)</strong
            ></mark
          ><br />
        </li>
      </ol>
      <script
        src="https://cdnjs.cloudflare.com/ajax/libs/prism/1.29.0/prism.min.js"
        integrity="sha512-7Z9J3l1+EYfeaPKcGXu3MS/7T+w19WtKQY/n+xzmw4hZhJ9tyYmcUS+4QqAlzhicE5LAfMQSF3iFTK9bQdTxXg=="
        crossorigin="anonymous"
        referrerpolicy="no-referrer"
      ></script>
      <link
        rel="stylesheet"
        href="https://cdnjs.cloudflare.com/ajax/libs/prism/1.29.0/themes/prism.min.css"
        integrity="sha512-tN7Ec6zAFaVSG3TpNAKtk4DOHNpSwKHxxrsiw4GHKESGPs5njn/0sMCUMl2svV4wo4BK/rCP7juYz+zx+l6oeQ=="
        crossorigin="anonymous"
        referrerpolicy="no-referrer"
      />
      <pre
        class="code code-wrap"
      ><code class="language-Plain Text" style="white-space:pre-wrap;word-break:break-all">Timeline (120Hz example, 8.33ms per frame)
0ms     Event triggered (onClick etc.)
1ms     React scheduling update begins
        ├── render() executes, calculates new virtual DOM
        ├── DOM updates
        ├── useLayoutEffect (synchronous execution)
        └── useEffect (synchronous execution)
3ms     React update completes

8.33ms  Next frame begins, rAF callback executes
        └── Preparing rendering flow

9ms     Browser rendering begins
        ├── Style Calculation
        ├── Layout
        ├── Paint
        └── Composite</code></pre>
      <p class=""></p>
      <p class=""></p>
      <p class=""></p>
      <p class=""></p>
      <p class=""></p>
      <p class=""></p>
      <p class=""></p>
      <p class=""></p>
      <h1 class="">Event Timing and Lifecycle</h1>
      <p class="">
        The complete event sequence from HTML loading to page usability,
        traditionally understood as:
      </p>
      <p class=""></p>
      <p class="">
        HTML parsing → DOMContentLoaded → Framework initialization →
        requestAnimationFrame → First Paint → window.onload →
        requestIdleCallback
      </p>
      <script
        src="https://cdnjs.cloudflare.com/ajax/libs/prism/1.29.0/prism.min.js"
        integrity="sha512-7Z9J3l1+EYfeaPKcGXu3MS/7T+w19WtKQY/n+xzmw4hZhJ9tyYmcUS+4QqAlzhicE5LAfMQSF3iFTK9bQdTxXg=="
        crossorigin="anonymous"
        referrerpolicy="no-referrer"
      ></script>
      <link
        rel="stylesheet"
        href="https://cdnjs.cloudflare.com/ajax/libs/prism/1.29.0/themes/prism.min.css"
        integrity="sha512-tN7Ec6zAFaVSG3TpNAKtk4DOHNpSwKHxxrsiw4GHKESGPs5njn/0sMCUMl2svV4wo4BK/rCP7juYz+zx+l6oeQ=="
        crossorigin="anonymous"
        referrerpolicy="no-referrer"
      />
      <pre
        class="code code-wrap"
      ><code class="language-JavaScript" style="white-space:pre-wrap;word-break:break-all">  0ms  ─────┬─────────────────────────────────────────────────────────────
            │ Browser sends request, starts downloading HTML/CSS/JS/images
            │ ├─ 16ms frame loop begins, but may have no rAF callbacks
            │ └─ User screen: nothing
            │
            │ Parsing HTML → building DOM Tree
            │ Parsing CSS → building CSSOM
            │ Possible synchronous JS blocking (script src synchronous loading &amp; execution)
            │ ├─ If JS registers rAF, only then will callbacks execute
            │ └─ User screen: nothing
            │
            │ ┌───────────────┐
            │ │ DOMContentLoaded (about 100-300ms) │ ← DOM structure complete, but images not loaded
            │ └───────────────┘
            │ └─ User screen: nothing
            │
            │ Vue/React initialization, starts render → generates virtual DOM
            │ ├─ Framework starts calculating Virtual DOM
            │ └─ User screen: nothing
            │
            │ diffing + patch DOM (about 150-250ms)
            │ ├─ JS thread: compare virtual DOM differences, operate real DOM, insert nodes
            │ ├─ Browser rendering: DOM structure changes, waiting for paint
            │ ├─ Framework might start registering rAF callbacks
            │ ├─ Only now do actual rAF callbacks start executing
            │ └─ User screen: nothing
            │
            │ Vue beforeMount → mounted (about 200-400ms)
            │ ├─ Components might register more rAF after mounted
            │ ├─ JS thread: ✅ DOM inserted, ❗not yet painted
            │ └─ User screen: nothing
            │
            │ requestAnimationFrame callback execution (rAF) ← if registered
            │ ├─ JS thread: final update preparation
            │ └─ User screen: ⚠️ screen being generated
            │
            │ Paint &amp; Composite (about 200-500ms)
            │ ↓ First Paint (FP) occurs: screen starts showing content
            │ ├─ Browser rendering: Style → Layout → Paint → Composite
            │ └─ User screen: first batch of content
            │
            │ First Contentful Paint (FCP): first content element finishes painting (about 300-600ms)
            │ ├─ If animations or continuous updates exist, rAF callbacks continue executing
            │ └─ User screen: meaningful content appears
            │
            │ Web resources continue loading (images, iframes)
            │ ├─ Loading process might have animation effects, generating more rAF
            │ └─ User screen: content gradually enriches
            │
            │ Largest Contentful Paint (LCP) (about 1000-2500ms)
            │ ├─ By now if rAF was registered, already executed 60+ times
            │ └─ User screen: main content complete
            │
            │ window.onload event triggers (about 1200-3000ms) ← all resources loaded
            │ ├─ After onload might register new rAF callbacks
            │ ├─ JS thread: all resources processed
            │ └─ User screen: complete screen
            │
            │ requestIdleCallback (rIC) executes during idle time
            │
  3000ms ───┴─────────────────────────────────────────────────────────────
</code></pre>
      <p class=""></p>
      <p class="">But actual situations vary depending on website nature.</p>
      <p class="">
        Modern websites, due to heavy third-party script usage, often see DCL
        delayed until after visual content (FCP/LCP), making traditional DCL
        timing judgment unreliable.
      </p>
      <p class="">
        FCP → LCP → DCL → L (observable in DevTools &gt; Performance)
      </p>
      <p class="">
        DCL triggering needs to wait for all synchronous scripts and defer
        scripts to complete execution, while modern websites often load many
        third-party services that depend on DOM:
      </p>
      <script
        src="https://cdnjs.cloudflare.com/ajax/libs/prism/1.29.0/prism.min.js"
        integrity="sha512-7Z9J3l1+EYfeaPKcGXu3MS/7T+w19WtKQY/n+xzmw4hZhJ9tyYmcUS+4QqAlzhicE5LAfMQSF3iFTK9bQdTxXg=="
        crossorigin="anonymous"
        referrerpolicy="no-referrer"
      ></script>
      <link
        rel="stylesheet"
        href="https://cdnjs.cloudflare.com/ajax/libs/prism/1.29.0/themes/prism.min.css"
        integrity="sha512-tN7Ec6zAFaVSG3TpNAKtk4DOHNpSwKHxxrsiw4GHKESGPs5njn/0sMCUMl2svV4wo4BK/rCP7juYz+zx+l6oeQ=="
        crossorigin="anonymous"
        referrerpolicy="no-referrer"
      />
      <pre
        class="code code-wrap"
      ><code class="language-HTML" style="white-space:pre-wrap;word-break:break-all">  &lt;!-- These block DCL --&gt;
  &lt;script src=&quot;jquery.js&quot;&gt;&lt;/script&gt;
  &lt;script defer src=&quot;analytics.js&quot;&gt;&lt;/script&gt;
  &lt;script defer src=&quot;marketing-tools.js&quot;&gt;&lt;/script&gt;
  &lt;script defer src=&quot;ab-testing.js&quot;&gt;&lt;/script&gt;

  &lt;!-- This doesn&#x27;t block DCL --&gt;
  &lt;script async src=&quot;ads.js&quot;&gt;&lt;/script&gt;</code></pre>
      <p class="">
        <mark class="highlight-red"
          >defer strategy: needs to operate existing DOM elements</mark
        >, needs sequential execution, downloads in parallel with HTML parsing,
        but execution blocks DCL
      </p>
      <p class="">
        async strategy: independent functionality scripts. Complete modules
        themselves, like ads, tracking, chat tools etc., their failure
        doesn&#x27;t affect page core functionality, neither download nor
        execution blocks DCL
      </p>
      <p class=""></p>
      <p class="">
        Simply put: async script failure, page can still function normally;
        defer script<br />failure might affect page functionality.
      </p>
      <p class="">
        Additionally,<br /><code
          >&lt;link
          rel={&#x27;stylesheet&#x27;|&#x27;preload&#x27;|&#x27;prefetch&#x27;|&#x27;preconnect&#x27;|&#x27;icon&#x27;}</code
        >
        are unrelated to DCL
      </p>
      <h3 class="">Thread Division</h3>
      <table class="simple-table">
        <thead class="simple-table-header">
          <tr>
            <th class="simple-table-header-color simple-table-header">Event</th>
            <th class="simple-table-header-color simple-table-header">
              Thread
            </th>
            <th class="simple-table-header-color simple-table-header">
              Key Characteristics
            </th>
          </tr>
        </thead>
        <tbody>
          <tr>
            <td class="">DOMContentLoaded</td>
            <td class="">Main Thread</td>
            <td class="">DOM structure + scripts complete</td>
          </tr>
          <tr>
            <td class="">Vue mounted</td>
            <td class="">Main Thread</td>
            <td class="">Component inserted into DOM</td>
          </tr>
          <tr>
            <td class="">requestAnimationFrame</td>
            <td class="">Main Thread</td>
            <td class="">Last chance before rendering</td>
          </tr>
          <tr>
            <td class="">First Paint</td>
            <td class="">Paint/Compositor</td>
            <td class="">Actually drawn to screen</td>
          </tr>
          <tr>
            <td class="">window.onload</td>
            <td class="">Main Thread</td>
            <td class="">All resources complete</td>
          </tr>
        </tbody>
      </table>
      <h3 class="">Relationship between rAF, onload and First Paint?</h3>
      <p class="">requestAnimationFrame → First Paint → window.onload</p>
      <p class="">
        rAF doesn&#x27;t happen automatically - callback execution completely
        depends on whether code registers them. The 16.67ms frame loop
        continues, but only executes when callbacks are registered
      </p>
      <h3 class="">Relationship between CRP and rAF?</h3>
      <p class="">
        Multiple rAF callbacks might execute during CRP process (each frame
        might have them). CRP completion indicator is First Paint occurrence,
        not single frame completion.
      </p>
      <p class=""></p>
      <p class=""></p>
      <p class=""></p>
      <p class=""></p>
      <p class=""></p>
      <p class=""></p>
      <p class=""></p>
      <h1 class="">Additional: Debug Ways</h1>
      <p class="">
        <mark class="highlight-red"
          ><strong>Chrome DevTools Performance </strong></mark
        >tab is the most practical tool. You can see what Main Thread and
        Compositor Thread are doing separately. Especially file loading timing,
        SEO metric trigger points, and when DCL and Load events actually happen.
        This tool basically answers most &quot;why is my page so slow&quot;
        questions.
      </p>
      <p class="">
        <mark class="highlight-red"
          ><strong>Chrome DevTools Rendering &amp; Layers</strong></mark
        >
        checks layer issues. When you suspect certain CSS changes affect
        performance, you can directly see which elements get promoted to
        independent layers, and whether properties like
        <mark class="highlight-red">transform</mark>,
        <mark class="highlight-red">opacity</mark>,
        <mark class="highlight-red">will-change</mark> actually create new
        layers.
      </p>
      <p class=""></p>
      <p class=""><strong>Deep Analysis Tools</strong></p>
      <ul class="bulleted-list">
        <li style="list-style-type: disc">
          chrome://tracing/ provides more detailed thread information, grouped
          by process. But this tool has massive information overload - usually
          only used when Performance tab can&#x27;t show the problem.
        </li>
      </ul>
      <ul class="bulleted-list">
        <li style="list-style-type: disc">
          Chrome Task Manager is like Mac&#x27;s Activity Monitor
        </li>
      </ul>
      <p class=""></p>
      <p class=""><strong>Specific Scenario Tools</strong></p>
      <p class="">
        <mark class="highlight-red">Lighthouse</mark> mainly runs SEO and
        performance metrics. Though comprehensive, the data is quite general.
        Gives you rough scores and tells you what each metric can optimize.
      </p>
      <p class="">
        <mark class="highlight-red">chrome://discards/graph</mark> and
        <mark class="highlight-red">pstree (cli)</mark> show process and thread
        relationship diagrams, but rarely used in actual development, unless
        researching browser internal architecture.
      </p>
      <p class=""></p>
      <p class="">
        Most performance issues can be solved with the first two tools. Others
        are more for deep research situations.
      </p>
      <p class=""></p>
      <p class=""></p>
      <p class=""></p>
      <h1 class="">References</h1>
      <p class="">thrashing</p>
      <p class="">
        <a
          href="https://web.dev/articles/avoid-large-complex-layouts-and-layout-thrashing"
          >https://web.dev/articles/avoid-large-complex-layouts-and-layout-thrashing</a
        >
      </p>
      <p class="">
        <a href="https://gist.github.com/kellegous/4646014"
          >https://gist.github.com/kellegous/4646014</a
        >
      </p>
      <p class="">
        <a href="https://kellegous.com/j/2013/01/26/layout-performance/"
          >https://kellegous.com/j/2013/01/26/layout-performance/</a
        >
      </p>
      <p class="">
        <a
          href="https://web.dev/articles/avoid-large-complex-layouts-and-layout-thrashing"
          >https://web.dev/articles/avoid-large-complex-layouts-and-layout-thrashing</a
        >
      </p>
      <p class="">
        <a
          href="https://developer.mozilla.org/en-US/docs/Web/Performance/Guides/CSS_JavaScript_animation_performance"
          >https://developer.mozilla.org/en-US/docs/Web/Performance/Guides/CSS_JavaScript_animation_performance</a
        >
      </p>
      <p class=""></p>
      <p class="">
        <a href="https://web.dev/articles/animations-guide"
          >https://web.dev/articles/animations-guide</a
        >
      </p>
      <p class="">
        <a href="https://web.dev/articles/critical-rendering-path"
          >https://web.dev/articles/critical-rendering-path</a
        >
      </p>
      <p class=""></p>
      <p class="">≤ 16ms bc browser overhead</p>
      <figure>
        <div class="source">
          https://github.com/vasanthk/browser-rendering-optimization/blob/master/README.md
        </div>
      </figure>
      <p class="">
        <a href="https://developer.chrome.com/blog/inside-browser-part2"
          >https://developer.chrome.com/blog/inside-browser-part2</a
        >
      </p>
      <p class="">
        <a href="https://developer.chrome.com/blog/inside-browser-part1"
          >https://developer.chrome.com/blog/inside-browser-part1</a
        >
      </p>
      <p class="">
        <a href="https://developer.chrome.com/blog/inside-browser-part3"
          >https://developer.chrome.com/blog/inside-browser-part3</a
        >
      </p>
      <p class="">
        <a href="https://developer.chrome.com/blog/inside-browser-part4"
          >https://developer.chrome.com/blog/inside-browser-part4</a
        >
      </p>
      <p class="">
        <a
          href="https://developer.chrome.com/docs/chromium/renderingng-architecture"
          >https://developer.chrome.com/docs/chromium/renderingng-architecture</a
        >
      </p>
      <p class="">
        <a href="https://web.dev/articles/fcp">https://web.dev/articles/fcp</a>
      </p>
      <p class="">
        <a href="https://web.dev/articles/lcp">https://web.dev/articles/lcp</a>
      </p>
      <p class=""></p>
      <p class=""></p>
      <p class=""></p>
    </div>
  </article>
  <span class="sans" style="font-size: 14px; padding-top: 2em"></span>
</body>
~/
about
posts
frontbacknetworkoscloud
readings
css
bookmarks
archives
© 2024 jialin00.com Original content since 2022
And maybe its just slow involvement at first, but try to sort of creep your career in that direction, because if youre not being challenged, if youre not a little bit scared all the time, just a little bit, then youre not gonna improve. - The Myth of the Genius Programmer