RSC 의 핵심은 "이 코드가 서버에서만 도는가, 클라이언트에 흘러가는가?" 경계. 보안 / 번들 / 성능 모두 이 경계에 달림. 'use client' / 'use server' 디렉티브를 의식적으로.
📖 핵심 개념
기본 = Server Component. 'use client' 가 있으면 그 파일 + 그 자식이 클라이언트 번들.
Server Component 는 async 가능, fetch / DB 직접 접근 가능, but state/effect 불가.
Client Component 는 일반 React (state/effect/event), but 번들에 포함.
💻 코드 패턴
// app/page.tsx — Server Component (기본)
import{db}from'@/lib/db';exportdefaultasyncfunctionPage() {constusers=awaitdb.users.findAll();// 서버에서 직접
return<UserListusers={users}/>;}// components/Counter.tsx — Client Component
'use client';import{useState}from'react';exportfunctionCounter() {const[n,setN]=useState(0);return<buttononClick={()=>setN(n+1)}>{n}</button>;}
// Server Component 가 Client Component 에 props 전달 OK (직렬화 가능 값만)
exportdefaultasyncfunctionPage() {constdata=awaitfetchData();return<ClientChartdata={data}/>;// ✅ data 는 plain object
}
🤔 의사결정 기준
컴포넌트 종류
Server
Client
DB / 파일시스템 / API key 사용
✅
❌ (보안)
사용자 입력 / state / effect
❌
✅
큰 의존성 (markdown parser, syntax highlighter)
✅ (번들 안 들어감)
❌
useTheme / useRouter / 브라우저 API
❌
✅
한 번 렌더 후 안 변함
✅
❌
❌ 안티패턴
'use client' 파일 안에서 db / process.env 접근: 번들에 포함되어 클라이언트로 누출. 'server-only' 패키지 사용해 import 시 에러 발생시키기.
Server Component 에서 useState/useEffect: 컴파일 에러 — Server 에는 없음.
Client Component 에 not-serializable props (function, Date, class instance) 전달: 직렬화 실패. plain object 만.
'use client' 를 모든 파일에 default: 번들 폭증. 필요한 leaf 만.
Server Action 에서 검증 누락: 'use server' 함수는 클라이언트가 직접 호출. zod 검증 + 인증 체크 필수.
🤖 LLM 활용 힌트
Next.js App Router 컨텍스트 명시.
LLM이 'use client' 를 부모에 붙이면 자식 모두 client 됨 → leaf 까지 미루도록 요청.