레퍼런스
use(resource)
컴포넌트에서 Promise나 context와 같은 데이터를 참조하려면 use
를 사용합니다.
import { use } from 'react';
function MessageComponent({ messagePromise }) {
const message = use(messagePromise);
const theme = use(ThemeContext);
// ...
다른 React Hook과 달리 use
는 if
와 같은 조건문과 반복문 내부에서 호출할 수 있습니다. 다른 React Hook과 같이 use
는 컴포넌트 또는 Hook에서만 호출할 수 있습니다.
Promise와 함께 호출될 때 use
Hook은 Suspense
및 error boundaries와 통합됩니다. use
에 전달된 Promise가 pending 되는 동안 use
를 호출하는 컴포넌트는 suspend됩니다. use
를 호출하는 컴포넌트가 Suspense 경계로 둘러싸여 있으면 fallback 이 표시됩니다. Promise가 리졸브되면 Suspense fallback은 use
Hook이 반환한 컴포넌트로 대체됩니다. use
에 전달된 Promise가 reject 되면 가장 가까운 Error Boundary의 fallback이 표시됩니다.
매개변수
반환값
use
Hook은 Promise나 context에서 참조한 값을 반환합니다.
use
Hook은 컴포넌트나 Hook 내부에서 호출되어야 합니다.- 서버 컴포넌트에서 데이터를 fetch 할 때는
use
보다async
및await
를 사용합니다. async
및await
은await
이 호출된 시점부터 렌더링을 시작하는 반면use
는 데이터가 리졸브된 후 컴포넌트를 리렌더링합니다.- 클라이언트 컴포넌트에서 Promise를 생성하는 것보다 서버 컴포넌트에서 Promise를 생성하여 클라이언트 컴포넌트에 전달하는 것이 좋습니다. 클라이언트 컴포넌트에서 생성된 Promise는 렌더링할 때마다 다시 생성됩니다. 서버 컴포넌트에서 클라이언트 컴포넌트로 전달된 Promise는 리렌더링 전반에 걸쳐 안정적입니다. 예시 확인하기.
사용법
use
를 사용하여 context 참조하기
context가 use
에 전달되면 useContext
와 유사하게 작동합니다. useContext
는 컴포넌트의 최상위 수준에서 호출해야 하지만 use
는 if
와 같은 조건문이나 for
과 같은 반복문 내부에서 호출할 수 있습니다. use
는 유연하므로 useContext
보다 선호됩니다.
import { use } from 'react';
function Button() {
const theme = use(ThemeContext);
// ...
use
는 전달한 context의 context 값을 반환합니다. context 값을 결정하기 위해 React는 컴포넌트 트리를 검색하고 위에서 가장 가까운 context provider를 찾습니다.
context를 Button
에 전달하려면 Button
또는 상위 컴포넌트 중 하나를 context provider로 래핑합니다.
function MyPage() {
return (
<ThemeContext.Provider value="dark">
<Form />
</ThemeContext.Provider>
);
}
function Form() {
// ... 버튼 렌더링 ...
}
provider와 Button
사이에 얼마나 많은 컴포넌트가 있는지는 중요하지 않습니다. Form
내부의 어느 곳이든 Button
이 use(ThemeContext)
를 호출하면 "dark"
를 값으로 받습니다
useContext
와 달리 use
는 if
와 같은 조건문과 반복문 내부에서 호출할 수 있습니다.
function HorizontalRule({ show }) {
if (show) {
const theme = use(ThemeContext);
return <hr className={theme} />;
}
return false;
}
use
는 if
내부에서 호출되므로 context에서 조건부로 값을 참조할 수 있습니다.
import { createContext, use } from 'react'; const ThemeContext = createContext(null); export default function MyApp() { return ( <ThemeContext.Provider value="dark"> <Form /> </ThemeContext.Provider> ) } function Form() { return ( <Panel title="Welcome"> <Button show={true}>Sign up</Button> <Button show={false}>Log in</Button> </Panel> ); } function Panel({ title, children }) { const theme = use(ThemeContext); const className = 'panel-' + theme; return ( <section className={className}> <h1>{title}</h1> {children} </section> ) } function Button({ show, children }) { if (show) { const theme = use(ThemeContext); const className = 'button-' + theme; return ( <button className={className}> {children} </button> ); } return false }
서버에서 클라이언트로 데이터 스트리밍하기
서버 컴포넌트에서 클라이언트 컴포넌트로 Promise prop을 전달하여 서버에서 클라이언트로 데이터를 스트리밍할 수 있습니다.
import { fetchMessage } from './lib.js';
import { Message } from './message.js';
export default function App() {
const messagePromise = fetchMessage();
return (
<Suspense fallback={<p>waiting for message...</p>}>
<Message messagePromise={messagePromise} />
</Suspense>
);
}
클라이언트 컴포넌트는 prop으로 받은 Promise를 use
API 에 전달합니다.
Client Component는 서버 컴포넌트가 처음에 생성한 Promise에서 값을 읽을 수 있습니다.
// message.js
'use client';
import { use } from 'react';
export function Message({ messagePromise }) {
const messageContent = use(messagePromise);
return <p>Here is the message: {messageContent}</p>;
}
Message
는 Suspense
로 래핑되어 있으므로 Promise가 리졸브될 때까지 fallback이 표시됩니다. Promise가 리졸브되면 use
Hook이 값을 참조하고 Message
컴포넌트가 Suspense fallback을 대체합니다.
"use client"; import { use, Suspense } from "react"; function Message({ messagePromise }) { const messageContent = use(messagePromise); return <p>Here is the message: {messageContent}</p>; } export function MessageContainer({ messagePromise }) { return ( <Suspense fallback={<p>⌛Downloading message...</p>}> <Message messagePromise={messagePromise} /> </Suspense> ); }
Deep Dive
Promise는 서버 컴포넌트에서 클라이언트 컴포넌트로 전달될 수 있으며 use
API 를 통해 클라이언트 컴포넌트에서 리졸브됩니다. 또한 서버 컴포넌트에서 await
을 사용하여 Promise를 리졸브하고 데이터를 클라이언트 컴포넌트에 prop
으로 전달하는 방법도 존재합니다.
export default async function App() {
const messageContent = await fetchMessage();
return <Message messageContent={messageContent} />
}
하지만 서버 컴포넌트에서 await
을 사용하면 await
문이 완료될 때까지 렌더링이 차단됩니다. 서버 컴포넌트에서 클라이언트 컴포넌트로 Promise를 prop으로 전달하면 Promise가 서버 컴포넌트의 렌더링을 차단하는 것을 방지할 수 있습니다.
거부된 Promise 처리하기
경우에 따라 use
에 전달된 Promise가 거부될 수 있습니다. 거부된 프로미스를 처리하는 방법은 2가지가 존재합니다.
error boundary를 사용하여 오류 표시하기
Promise가 reject 될 때 오류를 표시하고 싶다면 error boundary를 사용합니다. error boundary를 사용하려면 use
API 를 호출하는 컴포넌트를 error boundary로 래핑합니다. use
에 전달된 Promise가 reject 되면 error boundary에 대한 fallback이 표시됩니다.
"use client"; import { use, Suspense } from "react"; import { ErrorBoundary } from "react-error-boundary"; export function MessageContainer({ messagePromise }) { return ( <ErrorBoundary fallback={<p>⚠️Something went wrong</p>}> <Suspense fallback={<p>⌛Downloading message...</p>}> <Message messagePromise={messagePromise} /> </Suspense> </ErrorBoundary> ); } function Message({ messagePromise }) { const content = use(messagePromise); return <p>Here is the message: {content}</p>; }
Promise.catch
를 사용하여 대체 값 제공하기
use
에 전달된 Promise가 거부될 때 대체 값을 제공하려면 Promise의 catch
메서드를 사용합니다.
import { Message } from './message.js';
export default function App() {
const messagePromise = new Promise((resolve, reject) => {
reject();
}).catch(() => {
return "no new message found.";
});
return (
<Suspense fallback={<p>waiting for message...</p>}>
<Message messagePromise={messagePromise} />
</Suspense>
);
}
Promise의 catch
메서드를 사용하려면 Promise 객체에서 catch
를 호출합니다. catch
는 오류 메시지를 인자로 받는 함수를 인수로 받습니다. catch
에 전달된 함수가 반환하는하는 값은 모두 Promise의 리졸브 값으로 사용됩니다.
트러블슈팅
“Suspense Exception: This is not a real error!”
React 컴포넌트 또는 hook 함수 외부에서, 혹은 try-catch 블록에서 use
를 호출하고 있는 경우입니다. try-catch 블록 내에서 use
를 호출하는 경우 컴포넌트를 error boundary로 래핑하거나 Promise의 catch
를 호출하여 에러를 발견하고 Promise를 다른 값으로 리졸브합니다. 예시 확인하기
React 컴포넌트나 Hook 함수 외부에서 use
를 호출하는 경우 use
호출을 React 컴포넌트나 Hook 함수로 이동합니다.
function MessageComponent({messagePromise}) {
function download() {
// ❌ `use`를 호출하는 함수가 컴포넌트나 hook이 아닙니다.
const message = use(messagePromise);
// ...
컴포넌트 클로저 외부에서 use
를 호출합니다. 여기서 use
를 호출하는 함수는 컴포넌트 또는 Hook입니다.
function MessageComponent({messagePromise}) {
// ✅ `use`가 컴포넌트에서 호출되고 있습니다.
const message = use(messagePromise);
// ...