hydrateRoot
는 이전에 react-dom/server
로 생성된 HTML 콘텐츠를 가진 브라우저 DOM 노드 안에 React 컴포넌트를 표시할 수 있게 해줍니다.
const root = hydrateRoot(domNode, reactNode, options?)
레퍼런스
hydrateRoot(domNode, reactNode, options?)
hydrateRoot
를 호출하여 이미 서버 환경에서 렌더링된 기존 HTML에 React를 “붙여넣기” 합니다.
import { hydrateRoot } from 'react-dom/client';
const domNode = document.getElementById('root');
const root = hydrateRoot(domNode, reactNode);
React는 domNode
내부에 존재하는 HTML에 연결되어, 그 내부의 DOM 관리를 맡게 됩니다. React로 완전히 구축된 앱은 일반적으로 루트 컴포넌트와 함께 하나의 hydrateRoot
호출만 가집니다.
매개변수
-
domNode
: 서버에서 루트 요소Element로 렌더링된 DOM 요소. -
reactNode
: 기존 HTML에 렌더링하기 위한 “React 노드” 입니다. 주로ReactDOM Server
의renderToPipeableStream(<App />)
와 같은 메서드로 렌더링된<App />
과 같은 JSX 조각들입니다. -
optional
options
: React 루트에 대한 옵션을 가진 객체입니다.- optional
onCaughtError
: Callback called when React catches an error in an Error Boundary. Called with theerror
caught by the Error Boundary, and anerrorInfo
object containing thecomponentStack
. - optional
onUncaughtError
: Callback called when an error is thrown and not caught by an Error Boundary. Called with theerror
that was thrown and anerrorInfo
object containing thecomponentStack
. - optional
onRecoverableError
: React가 오류로부터 자동으로 복구될 때 호출되는 콜백. Called with theerror
React throws, and anerrorInfo
object containing thecomponentStack
. Some recoverable errors may include the original error cause aserror.cause
. - optional
identifierPrefix
: React가useId
에 의해 생성된 ID에 사용하는 문자열 접두사. 같은 페이지에서 여러개의 루트를 사용할 때 충돌을 피하는 데 유용합니다. 서버에서 사용한 값과 반드시 동일한 값이어야 합니다.
- optional
반환값
hydrateRoot
는 render
와 unmount
두 가지 메서드를 포함한 객체를 반환합니다.
주의 사항
hydrateRoot()
는 렌더링된 컨텐츠가 서버에서 렌더링된 컨텐츠와 동일할 것을 기대합니다. 따라서 불일치 사항은 버그로 취급하고 수정해야 합니다.- 개발 모드에서는 React가 Hydration 중 불일치에 대해 경고합니다. 불일치가 발생할 경우 속성 차이가 수정될 것이라는 보장은 없습니다. 이는 성능상의 이유로 중요한데, 대부분의 앱에서 불일치는 드물기 때문에 모든 마크업을 검증하는 것은 매우 비효율적이기 때문입니다.
- 앱에서
hydrateRoot
호출이 단 한번만 있을 가능성이 높습니다. 프레임워크를 사용한다면, 프레임워크가 이 호출을 대신 수행할 수도 있습니다. - 앱을 사전에 렌더링된 HTML 없이 클라이언트에서 직접 렌더링한다면,
hydrateRoot()
는 지원되지 않습니다.createRoot()
를 대신 사용해주세요.
root.render(reactNode)
브라우저 DOM 요소 내에서 Hydrate된 React 루트 안의 React 컴포넌트를 업데이트 하려면 root.render
를 호출하세요.
root.render(<App />);
React는 Hydrate된 root
에서 <App />
을 업데이트합니다.
매개변수
reactNode
: 업데이트하고 싶은 “React 노드”입니다. 주로<App />
같은 JSX를 매개변수로 넘기지만,createElement()
로 만든 React 요소 혹은 문자열, 숫자,null
,undefined
를 넘겨도 됩니다.
반환값
root.render
는 undefined
를 반환합니다.
주의 사항
- 루트가 Hydrate를 완료하기 전에
root.render
를 호출하면, React는 서버에서 렌더링된 HTML을 모두 없애고 클라이언트에서 렌더링된 컴포넌트들로 완전히 교체합니다.
root.unmount()
root.unmount
를 호출하면 React 루트 내부에서 렌더링된 트리를 삭제합니다.
root.unmount();
온전히 React만으로 작성된 앱에는 일반적으로 root.unmount
에 대한 호출이 없습니다.
이 함수는 주로 React 루트의 DOM 노드(또는 그 조상 노드)가 다른 코드에 의해 DOM에서 제거될 수 있는 경우에 유용합니다. 예를 들어 DOM에서 비활성 탭을 제거하는 jQuery 탭 패널을 상상해 보세요. 탭이 제거되면 그 안에 있는 모든 것(내부의 React 루트를 포함)이 DOM에서 제거됩니다. 이 경우 root.unmount
를 호출하여 제거된 루트의 콘텐츠 관리를 “중지”하도록 React에 지시해야 합니다. 그렇지 않으면 제거된 루트 내부의 컴포넌트는 구독과 같은 전역 리소스를 정리하고 확보하는 법을 모르는 채로 있게 됩니다.
root.unmount
를 호출하면 루트에 있는 모든 컴포넌트가 마운트 해제되고, 트리상의 이벤트 핸들러나 State가 제거되며, 루트 DOM 노드에서 React가 “분리”됩니다.
매개변수
root.unmount
는 매개변수를 받지 않습니다.
Returns
root.unmount
returns undefined
.
주의 사항
-
root.unmount
를 호출하면 트리의 모든 컴포넌트가 마운트 해제되고 루트 DOM 노드에서 React가 “분리”됩니다. -
root.unmount
를 한 번 호출한 후에는 같은 루트에서root.render
를 다시 호출할 수 없습니다. 마운트 해제된 루트에서root.render
를 호출하려고 하면 “마운트 해제된 루트를 업데이트할 수 없습니다.Cannot update an unmounted root” 오류가 발생합니다.
사용법
서버에서 렌더링된 HTML을 Hydrate하기
react-dom/server
로 앱의 HTML을 생성했다면, 클라이언트에서 Hydrate 해주어야 합니다.
import { hydrateRoot } from 'react-dom/client';
hydrateRoot(document.getElementById('root'), <App />);
위 코드를 통해 서버 HTML을 브라우저 DOM 노드에서 React 컴포넌트를 이용해 Hydrate 해줄 것 입니다. 주로 앱을 시작할 때 단 한 번 실행할 것입니다. 프레임워크를 사용중이라면 프레임워크가 대신 실행해 줄 것입니다.
앱을 Hydrate 하기 위해서 React는 컴포넌트의 로직을 사전에 서버에서 만들어 진 HTML에 “붙여넣을”것 입니다. Hydration을 통해 서버에서 만들어진 최초의 HTML 스냅샷을 브라우저에서 완전히 인터랙티브한 앱으로 바꿔주게 됩니다.
import './styles.css'; import { hydrateRoot } from 'react-dom/client'; import App from './App.js'; hydrateRoot( document.getElementById('root'), <App /> );
hydrateRoot
를 다시 호출하거나 다른 곳에서 더 호출할 필요는 없습니다. 이 시점부터 React가 애플리케이션의 DOM을 다루게 됩니다. 대신 UI를 갱신하기 위해선 State를 사용해야 합니다.
document
전체를 Hydrate하기
React로 앱을 모두 만들었을 경우 <html>
태그를 포함해 JSX로 된 전체 document
를 렌더링할 수 있습니다.
function App() {
return (
<html>
<head>
<meta charSet="utf-8" />
<meta name="viewport" content="width=device-width, initial-scale=1" />
<link rel="stylesheet" href="/styles.css"></link>
<title>My app</title>
</head>
<body>
<Router />
</body>
</html>
);
}
전체 document
를 Hydrate하기 위해선 전역 변수인 document
를 hydrateRoot
의 첫번째 인수로 넘깁니다.
import { hydrateRoot } from 'react-dom/client';
import App from './App.js';
hydrateRoot(document, <App />);
어쩔 수 없는 Hydration 불일치 오류 억제하기
어떤 요소Element의 속성이나 텍스트 컨텐츠가 서버와 클라이언트에서 어쩔 수 없이 다를 땐(예를 들어, timestamp를 이용했다거나), Hydration 불일치 경고를 안보이게 할 수 있습니다.
해당 요소에서 Hydration 경고를 끄기 위해선 suppressHydrationWarning={true}
를 추가하면 됩니다.
export default function App() { return ( <h1 suppressHydrationWarning={true}> Current Date: {new Date().toLocaleDateString()} </h1> ); }
이것은 한 단계 아래까지만 적용되며 탈출구Escape Hatch를 의도한 것입니다. 남용하지 마세요. 텍스트 컨텐츠가 아닌 한 React는 잘못된 부분을 수정하지 않을 것이며, 갱신이 일어나기 전까지는 불일치 상태로 남아있을 것입니다.
서로 다른 클라이언트와 서버 컨텐츠 다루기
의도적으로 서버와 클라이언트에서 서로 다른 내용을 렌더링하길 원한다면, 서버와 클라이언트에서 서로 다른 방법으로 렌더링하면 됩니다. 클라이언트에서 서버와는 다른 것을 렌더링할 때 클라이언트에선 Effect에서 true
로 할당되는 isClient
같은 State 변수를 읽을 수 있습니다.
import { useState, useEffect } from "react"; export default function App() { const [isClient, setIsClient] = useState(false); useEffect(() => { setIsClient(true); }, []); return ( <h1> {isClient ? 'Is Client' : 'Is Server'} </h1> ); }
이 방법은 처음엔 서버와 동일한 결과물을 렌더링하여 불일치 문제를 피하고, Hydration 후에 새로운 결과물이 동기적으로 렌더링됩니다.
Hydration된 루트 컴포넌트를 업데이트하기
루트의 Hydration이 끝난 후에, root.render
를 호출해 React 컴포넌트의 루트를 업데이트 할 수 있습니다. createRoot
와는 다르게 HTML로 최초의 컨텐츠가 이미 렌더링 되어 있기 때문에 자주 사용할 필요는 없습니다.
Hydration 후 어떤 시점에 root.render
를 호출한다면, 그리고 컴포넌트의 트리 구조가 이전에 렌더링했던 구조와 일치한다면, React는 State를 그대로 보존합니다. 입력 창Input에 어떻게 타이핑하든지 간에 문제가 발생하지 않습니다. 즉, 아래 예시에서처럼 매초 마다 상태를 업데이트하는 반복적인 render
를 문제 없이 렌더링 한다는 것을 알 수 있습니다.
import { hydrateRoot } from 'react-dom/client'; import './styles.css'; import App from './App.js'; const root = hydrateRoot( document.getElementById('root'), <App counter={0} /> ); let i = 0; setInterval(() => { root.render(<App counter={i} />); i++; }, 1000);
Hydration된 루트에서 root.render
를 호출하는 것은 흔한 일은 아닙니다. 내부 컴포넌트 중 한 곳에서 useState를 사용하는 것이 일반적입니다.
처리되지 않은 오류에 대한 대화 상자 표시하기
기본적으로 React는 처리되지 않은 모든 오류를 콘솔에 기록합니다. 자체적인 오류 보고 기능을 구현하려면 선택적 루트 옵션인 onUncaughtError
를 사용할 수 있습니다.
import { hydrateRoot } from 'react-dom/client';
const root = hydrateRoot(
document.getElementById('root'),
<App />,
{
onUncaughtError: (error, errorInfo) => {
console.error(
'Uncaught error',
error,
errorInfo.componentStack
);
}
}
);
root.render(<App />);
onUncaughtError 옵션은 두 개의 인수를 받는 함수입니다.
- 발생한 error.
- 오류의 componentStack을 포함하는 errorInfo 객체.
onUncaughtError
루트 옵션을 사용해 오류 대화 상자를 표시할 수 있습니다.
import { hydrateRoot } from "react-dom/client"; import App from "./App.js"; import {reportUncaughtError} from "./reportError"; import "./styles.css"; import {renderToString} from 'react-dom/server'; const container = document.getElementById("root"); const root = hydrateRoot(container, <App />, { onUncaughtError: (error, errorInfo) => { if (error.message !== 'Known error') { reportUncaughtError({ error, componentStack: errorInfo.componentStack }); } } });
Error Boundary 오류 표시하기
기본적으로 React는 Error Boundary에 의해 잡힌 모든 오류를 console.error
에 기록합니다. 이 동작을 재정의하려면 Error Boundary에서 잡힌 오류 처리에 대한 선택적 루트 옵션인 onCaughtError
를 사용할 수 있습니다.
import { hydrateRoot } from 'react-dom/client';
const root = hydrateRoot(
document.getElementById('root'),
<App />,
{
onCaughtError: (error, errorInfo) => {
console.error(
'Caught error',
error,
errorInfo.componentStack
);
}
}
);
root.render(<App />);
onCaughtError 옵션은 두 개의 인수를 받는 함수입니다.
- Error Boundary에 의해 잡힌 error.
- 오류의 componentStack을 포함하는 errorInfo.
onCaughtError
루트 옵션을 사용해 오류 대화 상자를 표시하거나 기록된 오류 중 알고 있는 오류를 필터링할 수 있습니다.
import { hydrateRoot } from "react-dom/client"; import App from "./App.js"; import {reportCaughtError} from "./reportError"; import "./styles.css"; const container = document.getElementById("root"); const root = hydrateRoot(container, <App />, { onCaughtError: (error, errorInfo) => { if (error.message !== 'Known error') { reportCaughtError({ error, componentStack: errorInfo.componentStack }); } } });
복구 가능한 Hydration 불일치 오류에 대한 대화 상자 표시하기
React가 Hydration 불일치를 만나면 클라이언트에서 자동으로 렌더링을 시도합니다. 기본적으로 React는 Hydration 불일치 오류를 console.error
에 기록합니다. 이 동작을 재정의하려면 선택적 루트 옵션인 onRecoverableError
를 사용할 수 있습니다.
import { hydrateRoot } from 'react-dom/client';
const root = hydrateRoot(
document.getElementById('root'),
<App />,
{
onRecoverableError: (error, errorInfo) => {
console.error(
'Caught error',
error,
error.cause,
errorInfo.componentStack
);
}
}
);
onRecoverableError 옵션은 두 개의 인수를 받는 함수입니다.
- React가 발생시킨 error. 일부 오류는 원래 원인을 error.cause에 포함하기도 합니다.
- 오류의 componentStack을 포함하는 errorInfo 객체.
Hydration 불일치에 대한 대화 상자를 표시하려면 onRecoverableError
루트 옵션을 사용할 수 있습니다.
import { hydrateRoot } from "react-dom/client"; import App from "./App.js"; import {reportRecoverableError} from "./reportError"; import "./styles.css"; const container = document.getElementById("root"); const root = hydrateRoot(container, <App />, { onRecoverableError: (error, errorInfo) => { reportRecoverableError({ error, cause: error.cause, componentStack: errorInfo.componentStack }); } });
문제 해결
다음과 같은 오류가 발생합니다: “You passed a second argument to root.render”
hydrateRoot
옵션을 root.render(...)
에 전달하는 실수가 흔히 일어나곤 합니다.
수정하려면 루트 옵션을 root.render(...)
대신 hydrateRoot(...)
에 전달하세요.
// 🚩 잘못된 방법: `root.render`는 하나의 인수만 받습니다.
root.render(App, {onUncaughtError});
// ✅ 올바른 방법: 옵션을 `createRoot`에 전달하세요.
const root = hydrateRoot(container, <App />, {onUncaughtError});