f8b21af4be
10_Wiki/Topics 대규모 정리: - 오류 캡처/미완성 stub 문서 227개 제거 - 교차폴더 중복 43클러스터 병합 (63파일 → redirect) - 링크명 정규화: 깨진 링크 수정·redirect 직결·개념 매핑 ~2,400건 - 카테고리 MOC 6개 신규 생성 - Graph 섹션 미해결 related-keyword 링크 10,058건 제거 Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>
5.0 KiB
5.0 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-prop-drilling | Prop Drilling | 10_Wiki/Topics | verified | self |
|
none | A | 0.9 | applied |
|
2026-05-10 | pending |
|
Prop Drilling
매 한 줄
"매 props 의 깊은 component tree 통과". 매 parent 가 grandchild 에 data 전달하기 위해 매 intermediate component 가 props 의 단순 forward. 2026 React 19+ 매 Server Component / Context / Zustand 의 5가지 해결책으로 회피.
매 핵심
매 문제
- Intermediate component 의 매 props type 의 noise.
- Refactoring 의 매 cost — 새 prop 추가 시 매 layer 수정.
- Type drift: 매 layer 마다 prop spread, original type 잃음.
매 5가지 해결
- Component Composition (children prop): 매 first choice.
- React Context: 매 theme, auth, locale — 매 cross-cutting concern.
- State Management Library: 매 Zustand, Jotai, Redux Toolkit.
- React Server Components (RSC): 매 server-side data fetching, props 의 root level resolve.
- Custom Hook + Module-level store: 매 small project.
매 응용
- Theme propagation: 매 Context.
- Auth user: 매 Context + custom hook (
useUser()). - Form state: 매 react-hook-form, TanStack Form.
- Server data: 매 RSC 또는 TanStack Query.
💻 패턴
Anti: 매 prop drilling
// App → Layout → Sidebar → UserMenu 매 user 4 level 통과
function App() {
const user = useAuth();
return <Layout user={user} />;
}
function Layout({ user }: { user: User }) {
return <Sidebar user={user} />;
}
function Sidebar({ user }: { user: User }) {
return <UserMenu user={user} />;
}
function UserMenu({ user }: { user: User }) {
return <div>{user.name}</div>;
}
해결 1: Composition
function App() {
const user = useAuth();
return (
<Layout sidebar={<Sidebar><UserMenu user={user} /></Sidebar>} />
);
}
function Layout({ sidebar }: { sidebar: ReactNode }) {
return <div>{sidebar}</div>;
}
해결 2: Context
const UserContext = createContext<User | null>(null);
function App() {
const user = useAuth();
return (
<UserContext.Provider value={user}>
<Layout />
</UserContext.Provider>
);
}
function UserMenu() {
const user = useContext(UserContext);
return <div>{user?.name}</div>;
}
// Custom hook for safety
function useUser() {
const user = useContext(UserContext);
if (!user) throw new Error('useUser must be inside UserProvider');
return user;
}
해결 3: Zustand
import { create } from 'zustand';
const useUserStore = create<{ user: User | null; setUser: (u: User) => void }>((set) => ({
user: null,
setUser: (user) => set({ user }),
}));
function UserMenu() {
const user = useUserStore((s) => s.user);
return <div>{user?.name}</div>;
}
해결 4: React Server Component (Next.js 16)
// app/layout.tsx (server component)
async function Layout({ children }: { children: ReactNode }) {
const user = await getUser(); // server-side
return (
<div>
<Sidebar user={user} />
{children}
</div>
);
}
해결 5: Custom hook + module store
// 매 simple, no library
let _user: User | null = null;
const listeners = new Set<() => void>();
export function useUser() {
return useSyncExternalStore(
(cb) => { listeners.add(cb); return () => listeners.delete(cb); },
() => _user,
);
}
export function setUser(u: User) {
_user = u;
listeners.forEach((l) => l());
}
매 결정 기준
| 상황 | 해결 |
|---|---|
| 1-2 level only | 그냥 props (drilling 아님) |
| Cross-cutting (theme, auth, locale) | Context |
| Frequently-changing state shared | Zustand / Jotai |
| Server-fetched data | RSC + TanStack Query |
| Complex slot-based UI | Composition (children) |
| Nested form | react-hook-form FormProvider |
기본값: 매 Composition 우선 → 매 Context (cross-cutting) → 매 Zustand (frequently-changing client state).
🔗 Graph
- 부모: React
- 변형: Zustand · Jotai · Redux_Toolkit
- Adjacent: Render_Props
🤖 LLM 활용
언제: refactor stage 매 component tree 의 prop forwarding 발견 시. 언제 X: 매 1-2 level 의 props — 매 over-engineering 의 risk.
❌ 안티패턴
- Context 남용: 매 every shared state — re-render storm.
- Single global Context: 매 split context for performance (theme separate from user).
- Redux for everything: 매 server state 매 TanStack Query 가 적합.
- Spread
{...props}의 chain: type 의 erosion + unused prop forward.
🧪 검증 / 중복
- Verified (React 19 docs, Kent C. Dodds blog, Zustand 5).
- 신뢰도 A.
🕓 Changelog
| 날짜 | 변경 |
|---|---|
| 2026-05-08 | Phase 1 |
| 2026-05-10 | Manual cleanup — 5가지 해결 + RSC 추가 |