5.3 KiB
5.3 KiB
id, title, category, status, source_trust_level, verification_status, created_at, updated_at, tags, tech_stack, applied_in, aliases
| id | title | category | status | source_trust_level | verification_status | created_at | updated_at | tags | tech_stack | applied_in | aliases | ||||||||||||||
|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|
| react-tanstack-query-advanced | TanStack Query 심화 — invalidate / optimistic / prefetch | Coding | draft | B | conceptual | 2026-05-09 | 2026-05-09 |
|
|
|
TanStack Query — Advanced
Server state 의 표준. queryKey 는 array + 안정, mutation 의
onMutate/onError/onSettled가 optimistic,setQueryData+invalidateQueries가 cache 동기화.
📖 핵심 개념
- queryKey: 캐시 식별 — 변경 시 새 쿼리.
- Stale: 데이터가 outdated 상태. Refetch trigger.
- Mutation: write —
onMutate로 cache 미리 갱신 (optimistic). - Hydration: SSR / RSC 의 dehydrated state 클라에 주입.
💻 코드 패턴
queryKey 표준화 (factories)
// queries/users.ts
export const userKeys = {
all: ['users'] as const,
lists: () => [...userKeys.all, 'list'] as const,
list: (filter: UserFilter) => [...userKeys.lists(), filter] as const,
details: () => [...userKeys.all, 'detail'] as const,
detail: (id: string) => [...userKeys.details(), id] as const,
};
useQuery({ queryKey: userKeys.detail(id), queryFn: () => api.user.get(id) });
queryClient.invalidateQueries({ queryKey: userKeys.lists() }); // 모든 list invalidate
Optimistic update
const qc = useQueryClient();
const mut = useMutation({
mutationFn: (patch: PostPatch) => api.post.update(patch.id, patch),
onMutate: async (patch) => {
// 1. 진행중 query cancel
await qc.cancelQueries({ queryKey: postKeys.detail(patch.id) });
// 2. 이전 값 백업
const prev = qc.getQueryData<Post>(postKeys.detail(patch.id));
// 3. 새 값으로 미리 set
qc.setQueryData<Post>(postKeys.detail(patch.id), old => ({ ...old!, ...patch }));
return { prev }; // context
},
onError: (err, patch, ctx) => {
if (ctx?.prev) qc.setQueryData(postKeys.detail(patch.id), ctx.prev);
},
onSettled: (_, __, patch) => {
qc.invalidateQueries({ queryKey: postKeys.detail(patch.id) });
},
});
Prefetch (Router/hover)
// route loader
await qc.prefetchQuery({ queryKey: userKeys.detail(id), queryFn: ... });
// hover
<Link to={`/u/${id}`} onMouseEnter={() => qc.prefetchQuery(...)}>...</Link>
Infinite query
const q = useInfiniteQuery({
queryKey: postKeys.lists(),
queryFn: ({ pageParam }) => api.post.list({ cursor: pageParam }),
initialPageParam: undefined,
getNextPageParam: (last) => last.nextCursor,
});
const items = q.data?.pages.flatMap(p => p.items) ?? [];
<button onClick={() => q.fetchNextPage()} disabled={!q.hasNextPage}>load more</button>
Suspense + ErrorBoundary
<Suspense fallback={<Spinner />}>
<ErrorBoundary fallbackRender={({ error, resetErrorBoundary }) => ...}>
<UserPanel id={id} />
</ErrorBoundary>
</Suspense>
function UserPanel({ id }) {
const { data } = useSuspenseQuery({ queryKey: userKeys.detail(id), queryFn: ... });
return <div>{data.name}</div>; // data 는 항상 정의됨
}
글로벌 default
const qc = new QueryClient({
defaultOptions: {
queries: {
staleTime: 30_000,
gcTime: 5 * 60_000,
retry: (count, err) => count < 2 && !isClientError(err),
refetchOnWindowFocus: 'always',
},
mutations: {
retry: 1,
},
},
});
SSR (Next.js)
// server component / loader
export default async function Page() {
const qc = new QueryClient();
await qc.prefetchQuery({ queryKey: userKeys.detail('1'), queryFn: ... });
return (
<HydrationBoundary state={dehydrate(qc)}>
<ClientPage id="1" />
</HydrationBoundary>
);
}
Dependent queries
const { data: user } = useQuery({ queryKey: userKeys.detail(id), queryFn: ... });
const { data: posts } = useQuery({
queryKey: postKeys.list({ userId: user?.id }),
queryFn: () => api.post.list({ userId: user!.id }),
enabled: !!user,
});
Selective updates (setQueriesData)
qc.setQueriesData<User[]>({ queryKey: userKeys.lists() }, (old) =>
old?.map(u => u.id === updated.id ? updated : u)
);
🤔 의사결정 기준
| 상황 | 권장 |
|---|---|
| Server state | TanStack Query |
| Client-only state | Zustand / useState |
| Form state | RHF |
| Realtime | useQuery + WebSocket invalidate |
| Pagination | useInfiniteQuery |
| 즉시 반응 UI | Optimistic update |
❌ 안티패턴
- queryKey unstable (객체 inline): 매 렌더 다른 key — 무한 fetch.
- Mutation 후 setQueryData 없이 invalidate 만: refetch 도착까지 빈 화면.
- Optimistic onError 복원 X: 실패 시 잘못된 cache.
- staleTime 0: 매 마운트 refetch — 과도.
- Global state 에 server data 복사: 두 source of truth.
- enabled false 인데 queryFn 무거움: ... 그래도 기본 안전.
- gcTime 너무 길음 (24h): 메모리 누적.
🤖 LLM 활용 힌트
- queryKey factory + invalidate hierarchy.
- Mutation 4단계 (cancel/backup/set/onSettled invalidate).
- SSR = HydrationBoundary.