79 lines
3.2 KiB
Markdown
79 lines
3.2 KiB
Markdown
---
|
|
id: react-rendering-optimization
|
|
title: React 렌더링 최적화 우선순위
|
|
category: Coding
|
|
status: draft
|
|
source_trust_level: B
|
|
verification_status: conceptual
|
|
created_at: 2026-05-09
|
|
updated_at: 2026-05-09
|
|
tags: [react, performance, profiler, vibe-coding]
|
|
tech_stack: { language: "TypeScript / React 18+", applicable_to: ["Web", "React Native"] }
|
|
applied_in: []
|
|
aliases: [memo, render perf, virtualization]
|
|
---
|
|
|
|
# React 렌더링 최적화 우선순위
|
|
|
|
> 측정 → 가장 큰 단일 원인 제거. **memo / useMemo / useCallback 부터 뿌리는 건 안티패턴**. 보통 진짜 원인은 (1) 거대 list 미가상화, (2) Context 광범위 broadcast, (3) 부모 state 변경에 따른 자식 cascade.
|
|
|
|
## 📖 핵심 개념
|
|
React Profiler 로 commit 시간 측정 → flame graph 에서 자주·길게 나오는 컴포넌트 찾기. 그 컴포넌트의 **재렌더 원인**을 끊는 게 핵심.
|
|
|
|
원인 4종:
|
|
1. props 참조 변경 (객체/함수 매번 새로)
|
|
2. Context value 변경
|
|
3. 부모 재렌더로 자식 자동 재렌더
|
|
4. 거대 list 의 모든 항목 재렌더
|
|
|
|
## 💻 코드 패턴
|
|
|
|
### 1. List 가상화 (가장 효과 큼)
|
|
```tsx
|
|
import { useVirtualizer } from '@tanstack/react-virtual';
|
|
// 1만 행 → 화면에 보이는 ~30개만 렌더
|
|
```
|
|
|
|
### 2. Context split
|
|
```tsx
|
|
// ❌ 한 Context 에 자주 바뀌는 값 + 안 바뀌는 값 같이
|
|
const AppContext = createContext({ user, theme, onlineUsers });
|
|
|
|
// ✅ 분리
|
|
const UserContext = createContext(...); // 거의 안 바뀜
|
|
const PresenceContext = createContext(...); // 자주 바뀜
|
|
```
|
|
|
|
### 3. State 위치 내리기
|
|
부모 state 변경이 모든 자식 재렌더 → state 를 진짜 쓰는 컴포넌트로 이동.
|
|
|
|
### 4. memoize 마지막 카드
|
|
```tsx
|
|
const Row = memo(({ item, onSelect }) => ...);
|
|
```
|
|
React.memo 가 효과 있으려면 props 가 안정 참조 → useCallback / useMemo 동반.
|
|
|
|
## 🤔 의사결정 기준 — 우선순위 순서
|
|
1. **Profiler 측정** (이거 안 했으면 멈춰)
|
|
2. List 가 길고 모두 렌더되면 → 가상화 (react-virtual / react-window)
|
|
3. Context value 가 자주 바뀌고 소비자 많으면 → context split
|
|
4. 부모 state 변경이 광범위 cascade → state 위치 재배치
|
|
5. 그 후에도 무거운 자식이 자주 재렌더 → React.memo + useCallback/useMemo
|
|
6. 그래도 느림 → 코드 분할 / lazy / Suspense
|
|
|
|
## ❌ 안티패턴
|
|
- **모든 컴포넌트에 React.memo**: 비교 비용 누적 + 안 변하는 props 도 비교. 측정 후 적용.
|
|
- **measure 없이 useMemo/useCallback 도배**: 메모리 + 비교만 늘리고 효과 0.
|
|
- **Context 에 무거운 값 + 자주 변경**: 모든 소비자 재렌더. split 또는 selector hook.
|
|
- **거대 list inline map**: 1만 행 매번. 가상화 안 하면 영원히 느림.
|
|
- **key={index}**: 항목 순서 바뀌면 React 가 잘못된 노드 재사용. stable id 필수.
|
|
|
|
## 🤖 LLM 활용 힌트
|
|
- 성능 문제 보고 시 "**Profiler 측정 결과 첨부, 그 후 우선순위 1→2→3 순으로 적용 권장**" 명시.
|
|
- LLM이 React.memo 부터 뿌리려고 하면 막기.
|
|
|
|
## 🔗 관련 문서
|
|
- [[React_useMemo_When_Not_To]]
|
|
- [[React_useCallback_Reality]]
|
|
- [[React_Context_API_Misuse]]
|