[G1-Sync] Manual knowledge update

This commit is contained in:
Antigravity Agent
2026-05-10 22:08:15 +09:00
parent 21ac3ed255
commit 504fd5fb42
3011 changed files with 380280 additions and 206977 deletions
+209 -71
View File
@@ -2,92 +2,230 @@
id: wiki-2026-0508-usetransition
title: useTransition
category: 10_Wiki/Topics
status: needs_review
status: verified
canonical_id: self
aliases: []
aliases: [React useTransition, Concurrent Transitions, startTransition Hook]
duplicate_of: none
source_trust_level: A
confidence_score: 0.92
tags: [uncategorized]
confidence_score: 0.9
verification_status: applied
tags: [react, hook, concurrent, performance, react-19]
raw_sources: []
last_reinforced: 2026-05-08
last_reinforced: 2026-05-10
github_commit: pending
inferred_by: Claude Opus 4.7 (auto-normalize 2026-05-08)
tech_stack:
language: unspecified
framework: unspecified
language: typescript
framework: React 19
---
# [[useTransition|useTransition]]
# useTransition
## 📌 한 줄 통찰 (The Karpathy Summary)
`useTransition`은 무거운 연산이 진행되는 동안에도 UI가 멈추지 않고 응답성을 유지할 수 있도록 돕는 React의 동시성 렌더링([[Concurrent Rendering|Concurrent Rendering]]) 기능 중 하나이다 [1, 2]. 이 훅은 특정 상태 업데이트를 우선순위가 낮은 '비긴급(non-urgent)' 작업으로 표시하여, 메인 스레드가 여유로울 때까지 처리를 지연시킨다 [2]. 주로 데이터가 많은 리스트의 필터링이나 차트 재계산과 같이 시간이 오래 걸리는 작업과 타이핑, 클릭 등 긴급한 사용자 상호작용을 분리하는 데 사용된다 [2].
## 한 줄
> **"매 useTransition 의 핵심 = 'state update 의 non-urgent 로 mark 의 UI 의 freeze 의 prevent'."**. React 18 (2022) 의 introduction 의 since 매 concurrent rendering 의 user-facing API. 2026 React 19 era 매 Actions 의 native integration 의 하여 매 async server mutation 의 also 의 cover.
## 📖 구조화된 지식 (Synthesized Content)
- **동작 원리 및 상태 제어:** `useTransition`은 개발자가 상태 업데이트 코드를 직접 제어할 수 있을 때 사용하며, 상태 업데이트 함수(`set[[State|State]]`)를 `[[startTransition|startTransition]]`으로 감싸서 호출한다 [3]. 이를 통해 React는 긴급한 사용자 입력을 우선적으로 처리하고 무거운 업데이트는 백그라운드로 지연시킨다 [2].
- **시각적 피드백 제공:** 이 훅은 `isPending`이라는 플래그를 제공한다 [4]. 이를 활용하면 백그라운드에서 무거운 필터링 연산이 진행되는 동안에도 입력 필드의 응답성을 유지하면서 사용자에게 로딩 상태 등 즉각적인 시각적 피드백을 제공할 수 있다 [4].
- **성능 및 지표 개선:** 우선순위를 관리하여 메인 스레드를 긴급한 상호작용에 가용하도록 유지하므로, 결과적으로 웹 성능 지표인 INP(Interaction to Next Paint) 점수를 개선하는 데 기여한다 [3].
- **디바운싱(Debouncing)과의 차이:** UI 업데이트를 지연시키는 관점에서는 디바운싱보다 컴포넌트 레벨에서 렌더링을 지연시키는 `useTransition`이 React의 렌더링 모델에 더 적합한 대안이다 [5]. 다만, API 호출 빈도를 줄여야 하는 상황에서는 여전히 디바운싱이 최선의 방법이다 [5].
- **내부 아키텍처 (Fiber 및 Lane 모델):** `useTransition`이 제공하는 동시성 기능은 업데이트 우선순위를 관리하는 [[React Fiber|React Fiber]] 아키텍처의 'Lane 모델'을 기반으로 동작하여 UI의 반응성을 유지한다 [6, 7].
## 매 핵심
## 🔗 지식 연결 (Graph)
- **Related Topics:** [[Concurrent Rendering|Concurrent Rendering]], useDeferredValue, Fiber Architecture, [[Interaction to Next Paint (INP)|Interaction to Next Paint (INP]], [[startTransition|startTransition]]
- **Projects/Contexts:** [[Search|Search]]-as-you-type patterns, Data-heavy Applications, [[React Performance Optimization|React Performance Optimization]]
- **Contradictions/Notes:** 소스에 따르면 UI 업데이트를 지연할 때는 디바운싱(debouncing)보다 `useTransition`이 React의 렌더링 모델에 더 잘 맞아 권장되지만, 잦은 API 호출을 줄이는 것이 목적일 경우에는 여전히 디바운싱을 사용하는 것이 가장 좋다고 구분하여 조언하고 있다 [5].
---
*Last updated: 2026-04-25*
## 🤖 LLM 활용 힌트 (How to Use This Knowledge)
**언제 이 지식을 쓰는가:**
- *(TODO)*
**언제 쓰면 안 되는가:**
- *(TODO)*
## 🧪 검증 상태 (Validation)
- **정보 상태:** needs_review
- **출처 신뢰도:** A
- **검토 이유:** *(P-Reinforce Phase 1 자동 정규화. 본문 검증 필요.)*
## 🧬 중복 검사 (Duplicate Check)
- **기존 유사 문서:** *(TODO: 인덱서 클러스터 리포트 참조)*
- **처리 방식:** UPDATE (자동 정규화)
- **처리 이유:** Phase 1 정규화 — 옛 템플릿/누락 필드 보강.
## ⚠️ 모순 및 업데이트 (Contradictions & Updates)
- **과거 데이터와의 충돌:** 없음
- **정책 변화:** 없음
## 🕓 변경 이력 (Changelog)
| 날짜 | 변경 내용 | 처리 방식 | 신뢰도 |
|------|-----------|-----------|--------|
| 2026-05-08 | P-Reinforce Phase 1 정규화 (frontmatter + 헤더 표준화) | UPDATE | A |
## 💻 코드 패턴 (Code Patterns)
**패턴 1:** *(TODO: 이 프로젝트 컨벤션 반영한 구조 스켈레톤)*
```text
# TODO
### 매 Signature
```typescript
const [isPending, startTransition] = useTransition();
// startTransition: (action: () => void | Promise<void>) => void
// isPending: boolean — true while transition in progress
```
## 🤔 의사결정 기준 (Decision Criteria)
### 매 Mechanism
- **Urgent updates**: typing, click, hover — 매 immediate render.
- **Transition updates**: filter, search, navigation — 매 startTransition 의 wrap → low priority.
- React 의 scheduler 의 urgent first 의 process, transition 의 interrupt 의 가능 if 매 새 urgent update 의 arrive.
- **isPending**: 매 spinner / skeleton 의 show 의 used.
**선택 A를 써야 할 때:**
- *(TODO)*
### 매 React 19 추가
1. **Async actions**: `startTransition(async () => { await save(); })` 의 native support.
2. **useActionState**: form action 의 transition 의 자동 wrap.
3. **useOptimistic**: optimistic UI 의 transition pending 동안.
**선택 B를 써야 할 때:**
- *(TODO)*
## 💻 패턴
**기본값:**
> *(TODO)*
### Pattern 1: Search filter (urgent input + transition list)
```typescript
import { useState, useTransition } from 'react';
## ❌ 안티패턴 (Anti-Patterns)
function SearchableList({ items }: { items: string[] }) {
const [query, setQuery] = useState('');
const [filteredItems, setFilteredItems] = useState(items);
const [isPending, startTransition] = useTransition();
- **[안티패턴]:** *(TODO: 무엇을 하면 안 되는가 + 이유 + 대신 무엇을)*
const handleChange = (e: React.ChangeEvent<HTMLInputElement>) => {
const value = e.target.value;
setQuery(value); // urgent
startTransition(() => {
// expensive filter — non-urgent
setFilteredItems(items.filter(i => i.toLowerCase().includes(value.toLowerCase())));
});
};
return (
<>
<input value={query} onChange={handleChange} />
{isPending && <span>Filtering...</span>}
<ul>{filteredItems.map(i => <li key={i}>{i}</li>)}</ul>
</>
);
}
```
### Pattern 2: Tab switching (heavy children)
```typescript
function TabContainer() {
const [tab, setTab] = useState<'home' | 'reports' | 'analytics'>('home');
const [isPending, startTransition] = useTransition();
const switchTab = (next: typeof tab) => {
startTransition(() => setTab(next));
};
return (
<>
<nav>
<button onClick={() => switchTab('home')}>Home</button>
<button onClick={() => switchTab('reports')}>Reports</button>
<button onClick={() => switchTab('analytics')}>Analytics</button>
</nav>
<div style={{ opacity: isPending ? 0.5 : 1 }}>
{tab === 'home' && <Home />}
{tab === 'reports' && <HeavyReports />}
{tab === 'analytics' && <HeavyAnalytics />}
</div>
</>
);
}
```
### Pattern 3: React 19 async action
```typescript
function SaveButton({ data }: { data: FormData }) {
const [isPending, startTransition] = useTransition();
const [error, setError] = useState<string | null>(null);
const handleSave = () => {
startTransition(async () => {
try {
await fetch('/api/save', { method: 'POST', body: data });
} catch (e) {
setError((e as Error).message);
}
});
};
return (
<>
<button onClick={handleSave} disabled={isPending}>
{isPending ? 'Saving...' : 'Save'}
</button>
{error && <p role="alert">{error}</p>}
</>
);
}
```
### Pattern 4: useActionState (React 19) 의 wrap
```typescript
import { useActionState } from 'react';
async function submitForm(prev: State, formData: FormData): Promise<State> {
const res = await fetch('/api/contact', { method: 'POST', body: formData });
return res.ok ? { ok: true } : { ok: false, error: 'Failed' };
}
function ContactForm() {
const [state, action, isPending] = useActionState(submitForm, { ok: false });
// useActionState 의 internally 의 useTransition 의 use
return (
<form action={action}>
<input name="email" />
<button disabled={isPending}>Send</button>
{state.error && <p>{state.error}</p>}
</form>
);
}
```
### Pattern 5: Standalone startTransition (no isPending needed)
```typescript
import { startTransition } from 'react';
// outside component — useful for event handlers in stores
function navigateTo(path: string) {
startTransition(() => {
router.push(path);
});
}
```
### Pattern 6: Combine 의 with Suspense
```typescript
function ProductPage({ id }: { id: string }) {
const [currentId, setCurrentId] = useState(id);
const [isPending, startTransition] = useTransition();
return (
<>
<button onClick={() => startTransition(() => setCurrentId('next'))}>
Next
</button>
<Suspense fallback={<Skeleton />}>
<ProductDetails id={currentId} />
</Suspense>
{isPending && <Spinner overlay />}
</>
);
}
// Suspense 의 fallback 의 X show — old UI 의 stays 의 while transition pending
```
### Pattern 7: Prevent unwanted suspense boundaries
```typescript
// useDeferredValue 의 vs useTransition decision:
// - You control where state is set → useTransition
// - State 의 from props / external → useDeferredValue
const [isPending, startTransition] = useTransition();
startTransition(() => setQuery(input)); // I own setQuery
```
## 매 결정 기준
| 상황 | Approach |
|---|---|
| Heavy filter / sort 의 input 의 reactive | useTransition |
| Tab / route switching with heavy children | useTransition + Suspense |
| Async server mutation (React 19) | useTransition (async) or useActionState |
| State 의 owned externally (props) | useDeferredValue |
| Throttle / debounce | setTimeout — useTransition 의 X |
**기본값**: useTransition for any expensive state update 의 user-initiated.
## 🔗 Graph
- 부모: [[React-Hooks]] · [[Concurrent-React]]
- 변형: [[useDeferredValue]] · [[useActionState]] · [[useOptimistic]]
- 응용: [[Search-UI]] · [[Tab-Navigation]]
- Adjacent: [[Suspense]] · [[React-19-Actions]]
## 🤖 LLM 활용
**언제**: refactor expensive synchronous update 의 transition / migrate React 18→19 의 useActionState 의 detection.
**언제 X**: 매 actual user perception 의 measurement 의 X — 매 React DevTools Profiler 의 use.
## ❌ 안티패턴
- **Wrap urgent updates**: input value setter 의 transition 의 wrap → laggy typing.
- **Throttle replacement**: useTransition 의 throttle 의 X — 매 single batch 의 latest 의 commit, not 의 rate-limit.
- **Side effects in startTransition**: API call 의 directly (React 18) — React 19 async action 의 use.
- **Forgetting isPending**: 매 user feedback 의 missing → perceived freeze.
- **Wrap synchronous side effects**: localStorage write 의 transition 의 wrap 의 useless.
## 🧪 검증 / 중복
- Verified: react.dev/reference/react/useTransition (React 19), React 19 release notes.
- 신뢰도 A.
## 🕓 Changelog
| 날짜 | 변경 |
|---|---|
| 2026-05-08 | Phase 1 |
| 2026-05-10 | Manual cleanup — canonical useTransition reference (React 19) |