Files
2nd/10_Wiki/Topics/AI_and_ML/L-component (Lifecycle Hooks).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

166 lines
5.0 KiB
Markdown

---
id: wiki-2026-0508-l-component-lifecycle-hooks
title: Lifecycle Hooks
category: 10_Wiki/Topics
status: verified
canonical_id: self
aliases: [Lifecycle, Component Lifecycle, useEffect, ngOnInit, onMount]
duplicate_of: none
source_trust_level: A
confidence_score: 0.9
verification_status: applied
tags: [frontend, react, vue, angular, svelte, lifecycle]
raw_sources: []
last_reinforced: 2026-05-10
github_commit: pending
tech_stack:
language: typescript
framework: react
---
# Lifecycle Hooks
## 매 한 줄
> **"매 컴포넌트의 생애주기 훅"**. mount/update/unmount 시점에 부수효과를 거는 표준 메커니즘. React 는 `useEffect` 단일화, Vue/Angular/Svelte 는 명명된 훅 제공.
## 매 핵심
### 매 공통 phase
1. **Create**: instance 생성, props 수신.
2. **Mount**: DOM 진입 — 이벤트 바인딩, 데이터 fetch.
3. **Update**: state/props 변경 — derived 갱신.
4. **Unmount**: DOM 제거 — cleanup (timer, listener, subscription).
5. **Error**: 자식 throw 잡기.
### 매 framework 매핑
| Phase | React | Vue 3 | Angular | Svelte |
|---|---|---|---|---|
| Mount | `useEffect(fn, [])` | `onMounted` | `ngOnInit` | `onMount` |
| Update | `useEffect(fn, [dep])` | `watch` | `ngOnChanges` | `$:` reactive |
| Unmount | return cleanup | `onUnmounted` | `ngOnDestroy` | `onDestroy` |
| Error | `ErrorBoundary` | `onErrorCaptured` | `ErrorHandler` | (없음, try/catch) |
### 매 React 19 메모
- StrictMode 가 `useEffect` 를 dev 에서 2회 실행 → cleanup 필수.
- Server Component 는 lifecycle 없음 (그냥 async function).
- `use` hook 으로 promise 직접 read 가능.
## 💻 패턴
### React: 마운트시 fetch + cleanup
```tsx
useEffect(() => {
const ctrl = new AbortController();
fetch(`/api/user/${id}`, { signal: ctrl.signal })
.then(r => r.json()).then(setUser);
return () => ctrl.abort();
}, [id]);
```
### React: subscribe pattern
```tsx
useEffect(() => {
const sub = store.subscribe(setState);
return () => sub.unsubscribe();
}, []);
```
### Vue 3 Composition API
```vue
<script setup lang="ts">
import { onMounted, onUnmounted, ref } from "vue";
const data = ref(null);
let timer: number;
onMounted(async () => {
data.value = await fetch("/api").then(r => r.json());
timer = setInterval(refresh, 5000);
});
onUnmounted(() => clearInterval(timer));
</script>
```
### Angular standalone component
```typescript
@Component({ selector: "app-x", standalone: true, template: "..." })
export class XComponent implements OnInit, OnDestroy {
private sub?: Subscription;
constructor(private svc: DataService) {}
ngOnInit() {
this.sub = this.svc.stream$.subscribe(v => (this.value = v));
}
ngOnDestroy() { this.sub?.unsubscribe(); }
}
```
### Svelte 5 (with runes)
```svelte
<script>
import { onMount, onDestroy } from "svelte";
let count = $state(0);
let timer;
onMount(() => { timer = setInterval(() => count++, 1000); });
onDestroy(() => clearInterval(timer));
</script>
```
### React custom hook (encapsulate lifecycle)
```tsx
function useInterval(cb: () => void, ms: number) {
const ref = useRef(cb);
useEffect(() => { ref.current = cb; }, [cb]);
useEffect(() => {
const id = setInterval(() => ref.current(), ms);
return () => clearInterval(id);
}, [ms]);
}
```
### Error boundary (React)
```tsx
class Boundary extends React.Component {
state = { err: null };
static getDerivedStateFromError(err) { return { err }; }
componentDidCatch(err, info) { logger.error(err, info); }
render() {
return this.state.err ? <Fallback /> : this.props.children;
}
}
```
## 매 결정 기준
| 작업 | 적절한 훅 (React) |
|---|---|
| 1회 fetch on mount | `useEffect(fn, [])` |
| dep 변경시 refetch | `useEffect(fn, [dep])` |
| DOM 측정 | `useLayoutEffect` |
| 외부 store 구독 | `useSyncExternalStore` |
| Render 동안 동기화 | derived state, **NOT** effect |
**기본값**: 가능하면 effect 안 쓰고 derived 로 처리. effect 는 "외부 시스템 동기화" 전용.
## 🔗 Graph
- 변형: [[useEffect]], [[ngOnInit]], [[onMount]]
- 응용: [[Data-Fetching]]
- Adjacent: [[Custom Hooks]], [[Error Boundaries]], [[Server Components]]
## 🤖 LLM 활용
**언제**: 외부 시스템(timer, socket, listener) 결합, mount-once init, prop-driven refetch.
**언제 X**: pure 계산 — useMemo/derived 로 충분, useEffect 남용은 안티패턴.
## ❌ 안티패턴
- **Effect 안 dep 누락**: stale closure → 버그 끝판왕.
- **Cleanup 미작성**: timer/listener leak, StrictMode 에서 즉시 들킴.
- **상태를 effect 로 동기화**: derived state 가 정답.
- **Async useEffect 함수**: `useEffect(async () => ...)` 안 됨. 안에서 async fn 호출.
- **ngOnChanges 에서 setState 무한 루프**: 입력 비교 필수.
## 🧪 검증 / 중복
- React docs (You Might Not Need an Effect), Vue 3 / Angular / Svelte 5 공식 문서.
- 신뢰도 A.
## 🕓 Changelog
| 날짜 | 변경 |
|---|---|
| 2026-05-08 | Phase 1 |
| 2026-05-10 | Manual cleanup — 4 framework 매핑표, useEffect 안티패턴, Svelte 5 runes |