๐ react-query๋ฅผ ์ด์ฉํ ์๋ฒ ๋ฐ์ดํฐ Prefetch ๋ฐฉ๋ฒ
Nextjs๋ ์๋ฒ์ฌ์ด๋ ๋ ๋๋ง์ ํตํด ์ฌ์ฉ์์๊ฒ ์์ฑ๋ HTML ํ์ด์ง๋ฅผ ๊ฑด๋ค์ค ์ ์์ต๋๋ค.
๊ทธ์ ๋๋ถ์ด ๋ฐ์ดํฐ fetching์ ํด๋ผ์ด์ธํธ๊ฐ ์๋ ์๋ฒ์์ ๋ฏธ๋ฆฌ ํ ๋ค, ๊ทธ ๊ฒฐ๊ณผ๋ฌผ์ HTML์ ํฌํจ์ํฌ ์๋ ์์ต๋๋ค.
์ด๋ ๊ฒ prefetching์ด ์ด๋ฃจ์ด์ง ๊ฒฝ์ฐ ๊ฑฐ์น๋ ๋คํธ์ํฌ ๋จ๊ณ๊ฐ ์ ์ด์ ธ ๋ ๋น ๋ฅด๊ณ , ์๋ฒ ์์์ ํธ์ถ-์๋ต์ด ์ด๋ฃจ์ด์ง๊ธฐ ๋๋ฌธ์ ๋ ์์ ํ๋ค๋ ์ด์ ์ ๋๋ฆด ์ ์์ต๋๋ค.
๊ทธ๋ผ Nextjs์์ react-query๋ฅผ ์ด์ฉํ prefetch ๋ฐฉ๋ฒ์ ์๊ฐํ๊ฒ ์ต๋๋ค.
๐ QueryProvider ์ธํ
react query๋ฅผ ์ฌ์ฉํ๊ธฐ ์ํด QueryClientProvider
๋ฅผ ์ํ ํด๋ผ์ด์ธํธ ์ฌ์ด๋ ์ปดํฌ๋ํธ๋ฅผ ๋ง๋ค์ด์ค ๋ค, layout.tsx
์ ๊ฐ์์ค๋๋ค.
// components/Providers.tsx
"use client";
import { QueryClient, QueryClientProvider } from "@tanstack/react-query";
function makeQueryClient() {
return new QueryClient();
}
let browserQueryClient: QueryClient | undefined = undefined;
function getQueryClient() {
if (typeof window === "undefined") {
return makeQueryClient();
} else {
if (!browserQueryClient) browserQueryClient = makeQueryClient();
return browserQueryClient;
}
}
export default function Providers({ children }: { children: React.ReactNode }) {
const queryClient = getQueryClient();
return (
<QueryClientProvider client={queryClient}>
{children}
</QueryClientProvider>
);
}
// app/layout.tsx
export default function RootLayout({
children,
}: Readonly<{
children: React.ReactNode;
}>) {
return (
<html lang="en">
<body className={inter.className}>
<div>
<Providers>{children}</Providers>
</div>
</body>
</html>
);
}
๋ ์์ธํ ์ธํ ๋ฐฉ๋ฒ์ ๋ค์ ๊ณต์ ๋ฌธ์๋ฅผ ์ฐธ๊ณ ํด์ฃผ์ธ์.
Server Rendering & Hydration | TanStack Query React Docs
This ad helps to keep us from burning out and rage-quitting OSS just *that* much more, so chill. ๐
tanstack.com
๐ ํธ์ถํ API Query ์ ๋ฆฌ
https://jsonplaceholder.typicode.com/
์ ๋ ์์ ๋ฅผ ์ํด ์์ mock api๋ฅผ ์ฌ์ฉํ์ต๋๋ค.
import { useQuery } from "@tanstack/react-query";
import axios from "axios";
type PostType = {
userId: number;
id: number;
title: string;
body: string;
};
export class PostService {
static async getPost(id: number) {
const response = await axios
.get<PostType>(`https://jsonplaceholder.typicode.com/posts/${id}`)
.then((data) => data);
return response.data;
}
static async getPostList() {
const response = await axios
.get<PostType[]>(`https://jsonplaceholder.typicode.com/posts`)
.then((data) => data);
return response.data;
}
}
export class PostQueryOption {
static getPost(id: number) {
return { queryKey: ["post", id], queryFn: () => PostService.getPost(id) };
}
static getPostList() {
return { queryKey: ["post"], queryFn: () => PostService.getPostList() };
}
}
export const usePost = (id: number) => {
return useQuery(PostQueryOption.getPost(id));
};
export const usePostList = () => {
return useQuery(PostQueryOption.getPostList());
};
Post๋ฅผ ๋จ์ผ๊ณผ ๋ฆฌ์คํธ๋ก ๋ถ๋ฌ์ค๋ API๋ฅผ ์ฌ์ฉํ์ต๋๋ค.
์ด๋ป๊ฒ ์ฌ์ฉ ๋ฐ ๋ถ๋ฆฌํด๋ ์๊ด์ ์์ง๋ง, ์ ๋ ํ ํ์ผ์ ๊ณตํต๋ ๊ด์ฌ์ฌ์ ๋ํ API ํธ์ถ ์ฝ๋ ๋ฐ ์ฟผ๋ฆฌ๋ฅผ ์์ ๊ฐ์ด ์ฌ์ ์ ์ํด ๋๊ณ ์ฌ์ฉํฉ๋๋ค.
๐ Prefetch ๊ณตํต ์ฝ๋ ๋ถ๋ฆฌ
import { PostQueryOption } from "./usePost";
import { PostList } from "./PostList";
import {
dehydrate,
HydrationBoundary,
QueryClient,
} from "@tanstack/react-query";
export default async function Page() {
const queryClient = new QueryClient();
await queryClient.prefetchQuery(PostQueryOption.getPostList());
return (
<HydrationBoundary state={dehydrate(queryClient)}>
<PostList />
</HydrationBoundary>
);
}
// PostList.tsx
"use client";
import Link from "next/link";
import { usePostList } from "./usePost";
export const PostList = () => {
const { data } = usePostList();
return (
<div>
{!!data?.length &&
data.map((item) => (
<Link key={item.id} href={`/post/${item.id}`}>
{item.title}
</Link>
))}
</div>
);
};
์๋ฒ์์ ํด๋ผ์ด์ธํธ ์ธก์ ๋ฐ์ดํฐ๋ฅผ ๋ณด๋ด๊ธฐ ์ ์ prefetch ํ๊ธฐ ์ํด์ ์์์ ์ฌ์ฉํ 3๊ฐ์ง ๊ธฐ๋ฅ์ ๋ํด ์ดํด๋ณด๊ฒ ์ต๋๋ค.
- queryClient.prefetchQuery
ํน์ ์ฟผ๋ฆฌ ๋ฐ์ดํฐ๋ฅผ ๋ฏธ๋ฆฌ ์๋ฒ์์ ๋ถ๋ฌ์์ ์บ์์ ์ ์ฅํ๋ ์ญํ ์ ํฉ๋๋ค.
queryClient.prefetchQuery๋ ๋ฐ์ดํฐ๋ฅผ ๋ฏธ๋ฆฌ ๊ฐ์ ธ์ค๋ ๊ธฐ๋ฅ์ ์ ๊ณตํ๋ ํจ์์ ๋๋ค. - HydrationBoundary
HydrationBoundary๋ ์๋ฒ์์ ๊ฐ์ ธ์จ ๋ฐ์ดํฐ๋ฅผ ํด๋ผ์ด์ธํธ์์ ์ฌ์ฌ์ฉํ ์ ์๋๋ก ์ํ๋ฅผ ํ์ด๋๋ ์ด์ ํ๋ ์ปดํฌ๋ํธ์ ๋๋ค. - dehydrate
์๋ฒ์์ ๊ฐ์ ธ์จ ๋ฐ์ดํฐ๋ฅผ ์ง๋ ฌํํ์ฌ ํด๋ผ์ด์ธํธ๋ก ์ ๋ฌํ๊ณ , ํด๋ผ์ด์ธํธ๊ฐ ํด๋น ๋ฐ์ดํฐ๋ฅผ ํ์ด๋๋ ์ด์ ํ ์ ์๋๋ก ๋์์ค๋๋ค.
์ ๊ธฐ๋ฅ๋ค์ ๋์ ๋ฐฉ์์ ์์ฝํ๋ฉด ๋ค์๊ณผ ๊ฐ์ต๋๋ค.
- ๋ฐ์ดํฐ ๋ฏธ๋ฆฌ ํจ์นญ
queryClient.prefetchQuery๋ฅผ ์ฌ์ฉํ์ฌ ์๋ฒ์์ ๊ฒ์๋ฌผ ๋ฆฌ์คํธ ๋ฐ์ดํฐ๋ฅผ ๋ฏธ๋ฆฌ ๋ถ๋ฌ์ ์บ์์ ์ ์ฅํฉ๋๋ค. - ๋ฐ์ดํฐ ์ง๋ ฌํ
dehydrate(queryClient)๋ฅผ ํตํด ์๋ฒ์์ ๋ฐ์์จ ๋ฐ์ดํฐ๋ฅผ ์ง๋ ฌํํ์ฌ ํด๋ผ์ด์ธํธ๋ก ์ ๋ฌํ ์ค๋น๋ฅผ ํฉ๋๋ค. - ํ์ด๋๋ ์ด์
ํด๋ผ์ด์ธํธ์์๋ HydrationBoundary๋ฅผ ์ฌ์ฉํ์ฌ ์๋ฒ์์ ๋ฏธ๋ฆฌ ๊ฐ์ ธ์จ ๋ฐ์ดํฐ๋ฅผ ์ฌ์ฌ์ฉํ์ฌ ๋ถํ์ํ ์ถ๊ฐ ๋ฐ์ดํฐ ํจ์นญ์ ๋ฐฉ์งํฉ๋๋ค.
๊ทธ๋ฌ๋ ์ ๋ฐฉ์๋๋ก ์ฌ์ฉํ๋ค๋ฉด prefetch๋ฅผ ์ฌ์ฉํ๊ธฐ ์ํ ์ค๋ณต ์ฝ๋๊ฐ ๋ฐ์ํ๊ธฐ ๋๋ฌธ์ ๋ค์๊ณผ ๊ฐ์ ํํ์ ์ปดํฌ๋ํธ๋ก ๋ถ๋ฆฌํ์ฌ ์ฌ์ฉํฉ๋๋ค.
import {
dehydrate,
FetchQueryOptions,
HydrationBoundary,
QueryClient,
} from "@tanstack/react-query";
type PrefetchBoundaryType = {
children: React.ReactElement;
options: FetchQueryOptions;
};
export default async function PrefetchBoundary({
children,
options,
}: PrefetchBoundaryType) {
const queryClient = new QueryClient();
await queryClient.prefetchQuery(options);
return (
<HydrationBoundary state={dehydrate(queryClient)}>
{children}
</HydrationBoundary>
);
}
// After
import { PostQueryOption } from "./usePost";
import { PostList } from "./PostList";
import PrefetchBoundary from "./PrefetchBoundary";
export default async function Page() {
return (
<PrefetchBoundary options={PostQueryOption.getPostList()}>
<PostList />
</PrefetchBoundary>
);
}
์ฌ๊ธฐ์ ์ฃผ์ํ ์ ์ prefetch ํ๋ ์ปดํฌ๋ํธ์ ๋ฐ๋์ async
ํค์๋๋ฅผ ์ถ๊ฐํ์ฌ ๋น๋๊ธฐ ๋ฐ์ดํฐ ํจ์นญ์ ์ํํ ํ์ ์ปดํฌ๋ํธ๋ฅผ ๋ ๋๋ง ํ๋๋ก ํฉ๋๋ค.
๋ง๋ฌด๋ฆฌํ๋ฉฐ prefetch๋ฅผ ์ฌ์ฉํด ๋ก๋ฉ ์๊ฐ์ ๋จ์ถํ๊ณ , ์ฌ์ฉ์ ๊ฒฝํ์ ๋ ๋์์ง๊ฒ ์๋ํด ๋ณด๊ฒ ์ต๋๋ค.
์ ์ฒด ์ฝ๋๋ ๊นํ๋ธ์์ ์ฐธ๊ณ ๋ฐ๋๋๋ค!
https://github.com/devlasbe/nextjs-playground/tree/main/app/2-prefetch
nextjs-playground/app/2-prefetch at main · LasBe-code/nextjs-playground
Nextjs ํธ๊ธฐ์ฌ ์ฒ๊ตญ. Contribute to LasBe-code/nextjs-playground development by creating an account on GitHub.
github.com
'React > Nextjs' ์นดํ ๊ณ ๋ฆฌ์ ๋ค๋ฅธ ๊ธ
[NextJS, SEO] AppRouter Dynamic Sitemap.xml ์์ฑํ๊ธฐ (0) | 2024.11.22 |
---|---|
[NextJS, SEO] AppRouter Dynamic Metadata ์์ฑํ๊ธฐ (0) | 2024.11.21 |
Vercel - Nextjs ๋ฐฉ๋ฌธ์ ์ ์ง๊ณ, ์น ๋ถ์ ์ด์ฉํ๊ธฐ (0) | 2024.10.10 |
[Nextjs] searchParams๋ก URL ์ฟผ๋ฆฌ ์คํธ๋ง ํ์ฑํ๊ธฐ (0) | 2024.09.09 |
[Nextjs] Dynamic Route, ๋์ ๋ผ์ฐํ (0) | 2024.08.03 |
๋๊ธ