useCallback 은 useMemo 의 함수 버전. 자식이 React.memo 거나 deps 에 함수가 들어가는 경우에만 의미 있음. 그 외엔 "옵티마이저처럼 보이는 코드 노이즈".
📖 핵심 개념
함수는 매 렌더마다 새 참조 생성 (JS object).
React.memo 자식은 props 참조 비교 → 새 함수 prop = 매번 재렌더.
useCallback 으로 안정화 → React.memo 효력 발휘.
useCallback 자체는 컴포넌트 렌더링을 최적화 안 한다. React.memo / useMemo / useEffect 의 deps 로 쓰일 때만 가치.
💻 코드 패턴
// ✅ React.memo 자식 + 함수 prop
constChild=memo(({onSelect}:{onSelect:(id: string)=>void})=>{...});functionParent() {constonSelect=useCallback((id: string)=>{...},[]);// deps 변하지 않으면 안정
return<ChildonSelect={onSelect}/>;}// ❌ memoized 안 된 자식 → 효력 없음 (어차피 매 렌더 새 인스턴스)
functionParent() {constonClick=useCallback(()=>{},[]);return<buttononClick={onClick}>X</button>;// 의미 없음
}
🤔 의사결정 기준
자식 / 사용처
useCallback 필요
React.memo 컴포넌트의 함수 prop
✅
useEffect / useMemo 의 deps 에 함수
✅
일반 DOM 핸들러 (onClick, onChange)
❌
한 번 이벤트 등록 후 떼는 핸들러
useEvent 패턴 또는 ref 보관
❌ 안티패턴
모든 함수에 useCallback: 대부분 무의미. 코드 노이즈.
React.memo 없는 자식 + useCallback: 효력 0.
deps 자주 바뀜: useCallback 가 매번 새 함수 → 안정화 효과 0.
stale closure: 함수가 state/props 캡처. deps 누락하면 옛 값 참조. exhaustive-deps lint 사용.
이벤트 핸들러 패턴 대신 useEvent 미사용: React 19 useEvent (실험적) 가 stale closure 문제 해결.