Files
2nd/10_Wiki/Topics/Architecture/Hydration.md
T
koriweb d8a80f6272 chore(wiki): dangling 링크 canonical 정규화 (768파일/1200건)
이름만 다른(표기 변형) [[위키링크]]를 대상 문서의 canonical 제목으로 치환해
끊겼던 1,200개 링크를 연결. 제목/파일명 정규화 일치만 적용하고 별칭 매칭은
과병합 위험으로 제외(애매성 가드). 원본은 _link_reconcile_backup/ 에 백업.
도구: Datacollect/scripts/link_reconcile_apply.mjs

Co-Authored-By: Claude Opus 4.8 <noreply@anthropic.com>
2026-06-08 12:24:15 +09:00

5.4 KiB

id, title, category, status, canonical_id, aliases, duplicate_of, source_trust_level, confidence_score, verification_status, tags, raw_sources, last_reinforced, github_commit, tech_stack
id title category status canonical_id aliases duplicate_of source_trust_level confidence_score verification_status tags raw_sources last_reinforced github_commit tech_stack
wiki-2026-0508-hydration Hydration 10_Wiki/Topics verified self
Client Hydration
SSR Hydration
none A 0.9 applied
web
ssr
react
frontend
2026-05-10 pending
language framework
typescript react

Hydration

매 한 줄

"매 server 가 render 한 static HTML 에 client JS 가 event handler / state 를 부여하는 과정". 매 SSR 의 SEO + first paint 이점 + SPA 의 interactivity 의 결합. 2026 시점 매 trend 는 매 less hydration — Astro Islands, React Server Components, Qwik resumability 가 매 full-page hydration 을 대체.

매 핵심

매 단계

  1. Server: render to HTML string (renderToString / renderToPipeableStream).
  2. Client: HTML parsing → first paint (TTFB / FCP).
  3. Hydration: JS download → React/framework 가 component tree 의 event listener attach + state restore.
  4. Interactive: TTI 도달.

매 cost

  • 매 download + parse + execute JS — 매 모바일 mid-tier 에서 무거움.
  • Hydration mismatch — 매 server vs client output 차이 시 warning + re-render.
  • 매 unused interactivity 도 매 hydrate (Wasted hydration).

매 modern alternatives

  • Selective / progressive hydration (React 18+): Suspense 단위로 lazy hydrate.
  • Islands architecture (Astro, Fresh): 매 interactive island 만 ship + hydrate.
  • Resumability (Qwik): hydration 자체 제거 — 매 server 에서 serialize 한 closure 를 매 event 시점에 lazy resume.
  • React Server Components: 매 server-only component 는 ship X — 매 client component 만 hydrate.

매 응용

  1. Next.js / Remix App Router (RSC + selective hydration).
  2. Astro (islands).
  3. Qwik / QwikCity (resumability).
  4. SvelteKit, SolidStart (fine-grained reactivity, hydration cheaper).

💻 패턴

React 19 SSR + hydrateRoot

// server.ts
import { renderToPipeableStream } from "react-dom/server";
import App from "./App";
const stream = renderToPipeableStream(<App/>, {
  bootstrapScripts: ["/client.js"],
  onShellReady() { stream.pipe(res); },
});

// client.ts
import { hydrateRoot } from "react-dom/client";
import App from "./App";
hydrateRoot(document, <App/>);

Suspense for selective hydration

<Suspense fallback={<Skeleton/>}>
  <HeavyChart/>{/* 매 이 boundary 만 lazy hydrate */}
</Suspense>

Astro Islands

---
import Counter from "../components/Counter.svelte";
---
<h1>Static</h1>
<Counter client:visible /> {/* 매 viewport 진입 시 hydrate */}

Qwik resumability

import { component$, useSignal } from "@builder.io/qwik";
export default component$(() => {
  const c = useSignal(0);
  // 매 click 전에는 JS 미실행 — 매 lazy resume
  return <button onClick$={() => c.value++}>{c.value}</button>;
});

Avoiding hydration mismatch

// 매 client-only value (Date.now, window) 는 useEffect 에서
function Now() {
  const [t, setT] = useState<string | null>(null);
  useEffect(() => setT(new Date().toISOString()), []);
  return <span>{t ?? ""}</span>; // 매 server: empty, client: 실제값
}

React Server Components (Next.js)

// app/page.tsx — server component, 매 hydrate X
export default async function Page() {
  const data = await db.posts.findMany();
  return <ClientWidget data={data}/>; // 매 ClientWidget 만 hydrate
}

Streaming SSR (Node)

const stream = renderToPipeableStream(<App/>, {
  onShellReady() { res.setHeader("content-type", "text/html"); stream.pipe(res); },
  onError(err) { console.error(err); },
});

매 결정 기준

상황 Approach
Mostly static + 적은 island Astro / 11ty + islands
Full app, dynamic data Next.js App Router (RSC)
Extreme TTI 우선 Qwik (resumability)
Fine-grained reactivity Svelte / Solid
매 SEO 불필요 SPA CSR (no hydration)

기본값: Next.js 15 App Router (RSC + selective hydration). 매 mostly-static 사이트 는 Astro.

🔗 Graph

🤖 LLM 활용

언제: TTI / LCP / INP 진단, framework 선택, hydration mismatch 디버깅. 언제 X: 매 SEO + first paint 둘 다 불필요한 internal tool — 매 CSR SPA 면 충분.

안티패턴

  • Hydration mismatch (Date.now, Math.random): 매 server-client output 불일치 → re-render.
  • Hydrating entire static page: 매 island 분리 안 함 → JS 무겁게.
  • Heavy useEffect in root: 매 hydration 직후 main thread block.
  • Conditional render based on typeof window: 매 mismatch 의 흔한 원인.
  • Hydrating below-the-fold immediately: 매 selective + visibility 활용.

🧪 검증 / 중복

  • Verified (React 19 docs, Next.js 15 docs, Astro docs, Qwik docs, Addy Osmani Hydration 2024).
  • 신뢰도 A.

🕓 Changelog

날짜 변경
2026-05-08 Phase 1
2026-05-10 Manual cleanup — hydration + islands + RSC + resumability 정리