useSyncExternalStore
useSyncExternalStore๋ ์ธ๋ถ store๋ฅผ ๊ตฌ๋
ํ ์ ์๋ React Hook์
๋๋ค.
const snapshot = useSyncExternalStore(subscribe, getSnapshot, getServerSnapshot?)๋ ํผ๋ฐ์ค
useSyncExternalStore(subscribe, getSnapshot, getServerSnapshot?)
์ปดํฌ๋ํธ์ ์ต์์ ๋ ๋ฒจ์์ useSyncExternalStore๋ฅผ ํธ์ถํ์ฌ ์ธ๋ถ ๋ฐ์ดํฐ ์ ์ฅ์์์ ๊ฐ์ ์ฝ์ต๋๋ค.
import { useSyncExternalStore } from 'react';
import { todosStore } from './todoStore.js';
function TodosApp() {
const todos = useSyncExternalStore(todosStore.subscribe, todosStore.getSnapshot);
// ...
}store์ ์๋ ๋ฐ์ดํฐ์ ์ค๋ ์ท์ ๋ฐํํฉ๋๋ค. ๋ ๊ฐ์ ํจ์๋ฅผ ์ธ์๋ก ์ ๋ฌํด์ผ ํฉ๋๋ค.
subscribeํจ์๋ store๋ฅผ ๊ตฌ๋ ํ๊ณ ๊ตฌ๋ ์ ์ทจ์ํ๋ ํจ์๋ฅผ ๋ฐํํด์ผ ํฉ๋๋ค.getSnapshotํจ์๋ store์์ ๋ฐ์ดํฐ์ ์ค๋ ์ท์ ์ฝ์ด์ผ ํฉ๋๋ค.
๋งค๊ฐ๋ณ์
-
subscribe: ํ๋์callback์ธ์๋ฅผ ๋ฐ์ store์ ๊ตฌ๋ ํ๋ ํจ์์ ๋๋ค. store๊ฐ ๋ณ๊ฒฝ๋ ๋, ์ ๊ณต๋callback์ด ํธ์ถ๋์ด React๊ฐgetSnapshot์ ๋ค์ ํธ์ถํ๊ณ (ํ์ํ ๊ฒฝ์ฐ) ์ปดํฌ๋ํธ๋ฅผ ๋ค์ ๋ ๋๋งํ๋๋ก ํด์ผ ํฉ๋๋ค.subscribeํจ์๋ ๊ตฌ๋ ์ ์ ๋ฆฌํ๋ ํจ์๋ฅผ ๋ฐํํด์ผ ํฉ๋๋ค. -
getSnapshot: ์ปดํฌ๋ํธ์ ํ์ํ store ๋ฐ์ดํฐ์ ์ค๋ ์ท์ ๋ฐํํ๋ ํจ์์ ๋๋ค. store๊ฐ ๋ณ๊ฒฝ๋์ง ์์ ์ํ์์getSnapshot์ ๋ฐ๋ณต์ ์ผ๋ก ํธ์ถํ๋ฉด ๋์ผํ ๊ฐ์ ๋ฐํํด์ผ ํฉ๋๋ค. ์ ์ฅ์๊ฐ ๋ณ๊ฒฝ๋์ด ๋ฐํ๋ ๊ฐ์ด ๋ค๋ฅด๋ฉด (Object.is์ ๋น๊ตํ์ฌ) React๋ ์ปดํฌ๋ํธ๋ฅผ ๋ฆฌ๋ ๋๋งํฉ๋๋ค. -
optional
getServerSnapshot: store์ ์๋ ๋ฐ์ดํฐ์ ์ด๊ธฐ ์ค๋ ์ท์ ๋ฐํํ๋ ํจ์์ ๋๋ค. ์๋ฒ ๋ ๋๋ง ๋์ค๊ณผ ํด๋ผ์ด์ธํธ์์ ์๋ฒ ๋ ๋๋ง ๋ ์ฝํ ์ธ ์ ํ์ด๋๋ ์ด์ ์ค์๋ง ์ฌ์ฉ๋ฉ๋๋ค. ์๋ฒ ์ค๋ ์ท์ ํด๋ผ์ด์ธํธ์ ์๋ฒ ๊ฐ์ ๋์ผํด์ผ ํ๋ฉฐ ์ผ๋ฐ์ ์ผ๋ก ์ง๋ ฌํ๋์ด ์๋ฒ์์ ํด๋ผ์ด์ธํธ๋ก ์ ๋ฌ๋ฉ๋๋ค. ์ด ํจ์๊ฐ ์ ๊ณต๋์ง ์์ผ๋ฉด ์๋ฒ์์ ์ปดํฌ๋ํธ๋ฅผ ๋ ๋๋งํ ๋ ์ค๋ฅ๊ฐ ๋ฐ์ํฉ๋๋ค.
๋ฐํ๊ฐ
๋ ๋๋ง ๋ก์ง์ ์ฌ์ฉํ ์ ์๋ store์ ํ์ฌ ์ค๋ ์ท์ ๋๋ค.
์ฃผ์ ์ฌํญ
-
getSnapshot์ด ๋ฐํํ๋ store ์ค๋ ์ท์ ๋ถ๋ณ์ด์ด์ผ ํฉ๋๋ค. ๊ธฐ๋ณธ ์คํ ์ด์ ๋ณ๊ฒฝ ๊ฐ๋ฅํ ๋ฐ์ดํฐ๊ฐ ์๋ ๊ฒฝ์ฐ ๋ฐ์ดํฐ๊ฐ ๋ณ๊ฒฝ๋ ๊ฒฝ์ฐ ๋ณ๊ฒฝ ๋ถ๊ฐ๋ฅํ ์ ์ค๋ ์ท์ ๋ฐํํฉ๋๋ค. ๊ทธ๋ ์ง ์์ผ๋ฉด ์บ์ ๋ ๋ง์ง๋ง ์ค๋ ์ท์ ๋ฐํํฉ๋๋ค. -
๋ฆฌ๋ ๋๋งํ๋ ๋์ ๋ค๋ฅธ
subscribeํจ์๊ฐ ์ ๋ฌ๋๋ฉด React๋ ์๋ก ์ ๋ฌ๋subscribeํจ์๋ฅผ ์ฌ์ฉํ์ฌ ์ ์ฅ์๋ฅผ ๋ค์ ๊ตฌ๋ ํฉ๋๋ค. ์ปดํฌ๋ํธ ์ธ๋ถ์์subscribe๋ฅผ ์ ์ธํ๋ฉด ์ด๋ฅผ ๋ฐฉ์งํ ์ ์์ต๋๋ค. -
non-blocking transition ์ ๋ฐ์ดํธ ์ค์ ์คํ ์ด๊ฐ ๋ณ๊ฒฝ๋๋ฉด, React๋ ํด๋น ์ ๋ฐ์ดํธ๋ฅผ blocking์ผ๋ก ์ํํ๋๋ก ๋๋์๊ฐ๋๋ค. ๊ตฌ์ฒด์ ์ผ๋ก, ๋ชจ๋ Transition ์ ๋ฐ์ดํธ์ ๋ํด React๋ DOM์ ๋ณ๊ฒฝ ์ฌํญ์ ์ ์ฉํ๊ธฐ ์ง์ ์
getSnapshot์ ํ ๋ฒ ๋ ํธ์ถํฉ๋๋ค. ์ฒ์ ํธ์ถํ์ ๋์ ๋ค๋ฅธ ๊ฐ์ ๋ฐํํ๋ฉด, React๋ ์ฒ์๋ถํฐ ๋ค์ ์ ๋ฐ์ดํธ๋ฅผ ์์ํ๊ณ , ์ด๋ฒ์๋ blocking ์ ๋ฐ์ดํธ๋ฅผ ์ ์ฉํ์ฌ ํ๋ฉด์ ๋ชจ๋ ์ปดํฌ๋ํธ๊ฐ ๊ฐ์ ์คํ ์ด ๋ฒ์ ์ ๋ฐ์ํ๋๋ก ํฉ๋๋ค. -
useSyncExternalStore๊ฐ ๋ฐํํ ์คํ ์ด ๊ฐ์ ๊ธฐ๋ฐ์ผ๋ก ๋ ๋๋ง์ ์ผ์ ์ค๋จ ํ๋ ๊ฒ์ ๊ถ์ฅํ์ง ์์ต๋๋ค. ๊ทธ ์ด์ ๋ ์ธ๋ถ ์คํ ์ด์ ๋ํ ๋ณํ์ non-blocking transition ์ ๋ฐ์ดํธ๋ก ํ์ํ ์ ์๊ธฐ ๋๋ฌธ์, ๊ฐ์ฅ ๊ฐ๊น์ดSuspensefallback์ ํธ๋ฆฌ๊ฑฐํด์, ํ๋ฉด์์ ๋ก๋ฉ ์คํผ๋๋ก ๋์ฒดํ์ฌ, ์ผ๋ฐ์ ์ผ๋ก UX๊ฐ ์ข์ง ์๊ธฐ ๋๋ฌธ์ ๋๋ค.์๋ฅผ ๋ค์ด, ๋ค์์ ๊ถ์ฅ๋์ง ์์ต๋๋ค.
const LazyProductDetailPage = lazy(() => import('./ProductDetailPage.js'));function ShoppingApp() {const selectedProductId = useSyncExternalStore(...);// โ `selectedProductId`์ ์ข ์๋ Promise๋ก `use`๋ฅผ ํธ์ถํ๋ ๊ฒconst data = use(fetchItem(selectedProductId))// โ `selectedProductId`๋ฅผ ๊ธฐ๋ฐ์ผ๋ก ์ง์ฐ ์ปดํฌ๋ํธ๋ฅผ ์กฐ๊ฑด๋ถ๋ก ๋ ๋๋งํ๋ ๊ฒreturn selectedProductId != null ? <LazyProductDetailPage /> : <FeaturedProducts />;}
์ฌ์ฉ๋ฒ
์ธ๋ถ store ๊ตฌ๋
๋๋ถ๋ถ์ React ์ปดํฌ๋ํธ๋ props, state, ๊ทธ๋ฆฌ๊ณ context์์๋ง ๋ฐ์ดํฐ๋ฅผ ์ฝ์ต๋๋ค. ํ์ง๋ง ๋๋ก๋ ์ปดํฌ๋ํธ๊ฐ ์๊ฐ์ด ์ง๋จ์ ๋ฐ๋ผ ๋ณ๊ฒฝ๋๋ React ์ธ๋ถ์ ์ผ๋ถ ์ ์ฅ์์์ ์ผ๋ถ ๋ฐ์ดํฐ๋ฅผ ์ฝ์ด์ผ ํ๋ ๊ฒฝ์ฐ๊ฐ ์์ต๋๋ค. ๋ค์์ด ํฌํจ๋ฉ๋๋ค.
- React ์ธ๋ถ์ state๋ฅผ ๋ณด๊ดํ๋ ์๋ํํฐ ์ํ ๊ด๋ฆฌ ๋ผ์ด๋ธ๋ฌ๋ฆฌ.
- ๋ณ๊ฒฝ ๊ฐ๋ฅํ ๊ฐ์ ๋ ธ์ถํ๋ ๋ธ๋ผ์ฐ์ API์ ๊ทธ ๋ณ๊ฒฝ ์ฌํญ์ ๊ตฌ๋ ํ๋ ์ด๋ฒคํธ.
์ธ๋ถ ๋ฐ์ดํฐ ์ ์ฅ์์์ ๊ฐ์ ์ฝ์ผ๋ ค๋ฉด ์ปดํฌ๋ํธ์ ์ต์์ ๋ ๋ฒจ์์ useSyncExternalStore๋ฅผ ํธ์ถํ์ธ์.
import { useSyncExternalStore } from 'react';
import { todosStore } from './todoStore.js';
function TodosApp() {
const todos = useSyncExternalStore(todosStore.subscribe, todosStore.getSnapshot);
// ...
}store์ ์๋ ๋ฐ์ดํฐ์ snapshot์ ๋ฐํํฉ๋๋ค. ๋ ๊ฐ์ ํจ์๋ฅผ ์ธ์๋ก ์ ๋ฌํด์ผ ํฉ๋๋ค.
subscribeํจ์๋ store์ ๊ตฌ๋ ํ๊ณ ๊ตฌ๋ ์ ์ทจ์ํ๋ ํจ์๋ฅผ ๋ฐํํด์ผ ํฉ๋๋ค.getSnapshotํจ์๋ store์์ ๋ฐ์ดํฐ์ ์ค๋ ์ท์ ์ฝ์ด์ผ ํฉ๋๋ค.
React๋ ์ด ํจ์๋ฅผ ์ฌ์ฉํด ์ปดํฌ๋ํธ๋ฅผ store์ ๊ตฌ๋ ํ ์ํ๋ก ์ ์งํ๊ณ ๋ณ๊ฒฝ ์ฌํญ์ด ์์ ๋ ๋ฆฌ๋ ๋๋งํฉ๋๋ค.
์๋ฅผ ๋ค์ด ์๋ ์๋๋ฐ์ค์์ todosStore๋ React ์ธ๋ถ์ ๋ฐ์ดํฐ๋ฅผ ์ ์ฅํ๋ ์ธ๋ถ store๋ก ๊ตฌํ๋์ด ์์ต๋๋ค. TodosApp์ปดํฌ๋ํธ๋ useSyncExternalStore Hook์ผ๋ก ํด๋น ์ธ๋ถ store์ ์ฐ๊ฒฐํฉ๋๋ค.
import { useSyncExternalStore } from 'react'; import { todosStore } from './todoStore.js'; export default function TodosApp() { const todos = useSyncExternalStore(todosStore.subscribe, todosStore.getSnapshot); return ( <> <button onClick={() => todosStore.addTodo()}>Add todo</button> <hr /> <ul> {todos.map(todo => ( <li key={todo.id}>{todo.text}</li> ))} </ul> </> ); }
๋ธ๋ผ์ฐ์ API ๊ตฌ๋
useSyncExternalStore๋ฅผ ์ถ๊ฐํ๋ ๋ ๋ค๋ฅธ ์ด์ ๋ ์๊ฐ์ด ์ง๋จ์ ๋ฐ๋ผ ๋ณ๊ฒฝ๋๋ ๋ธ๋ผ์ฐ์ ์ ๋
ธ์ถ๋๋ ์ผ๋ถ ๊ฐ์ ๊ตฌ๋
ํ๋ ค๋ ๊ฒฝ์ฐ์
๋๋ค. ์๋ฅผ ๋ค์ด ์ปดํฌ๋ํธ์ ๋คํธ์ํฌ ์ฐ๊ฒฐ์ด ํ์ฑํ๋์ด ์๋์ง ์ฌ๋ถ๋ฅผ ํ์ํ๊ณ ์ถ๋ค๊ณ ๊ฐ์ ํด ๋ณด๊ฒ ์ต๋๋ค. ๋ธ๋ผ์ฐ์ ๋ navigator.onLine.์ด๋ผ๋ ์์ฑ์ ํตํด ์ด ์ ๋ณด๋ฅผ ๋
ธ์ถํฉ๋๋ค.
์ด ๊ฐ์ ์๊ฐ์ด ์ง๋จ์ ๋ฐ๋ผ React๊ฐ ์์ง ๋ชปํ๋ ์ฌ์ด์ ๋ณ๊ฒฝ๋ ์ ์์ผ๋ฏ๋ก useSyncExternalStore๋ก ๊ฐ์ ์ฝ์ด์ผ ํฉ๋๋ค.
import { useSyncExternalStore } from 'react';
function ChatIndicator() {
const isOnline = useSyncExternalStore(subscribe, getSnapshot);
// ...
}getSnapshot ํจ์๋ฅผ ๊ตฌํํ๋ ค๋ฉด ๋ธ๋ผ์ฐ์ API์์ ํ์ฌ ๊ฐ์ ์ฝ์ต๋๋ค.
function getSnapshot() {
return navigator.onLine;
}๋ค์์ผ๋ก subscribe ํจ์๋ฅผ ๊ตฌํํด์ผ ํฉ๋๋ค. ์๋ฅผ ๋ค์ด navigator.onLine์ด ๋ณ๊ฒฝ๋๋ฉด ๋ธ๋ผ์ฐ์ ๋ window ๊ฐ์ฒด์์ online ๋ฐ offline ์ด๋ฒคํธ๋ฅผ ์คํํฉ๋๋ค. callback ์ธ์๋ฅผ ํด๋น ์ด๋ฒคํธ์ ๊ตฌ๋
ํ ๋ค์ ๊ตฌ๋
์ ์ ๋ฆฌํ๋ ํจ์๋ฅผ ๋ฐํํด์ผ ํฉ๋๋ค.
function subscribe(callback) {
window.addEventListener('online', callback);
window.addEventListener('offline', callback);
return () => {
window.removeEventListener('online', callback);
window.removeEventListener('offline', callback);
};
}์ด์ React๋ ์ธ๋ถ navigator.onLine API์์ ๊ฐ์ ์ฝ๋ ๋ฐฉ๋ฒ๊ณผ ๊ทธ ๋ณ๊ฒฝ ์ฌํญ์ ๊ตฌ๋
ํ๋ ๋ฐฉ๋ฒ์ ์๊ณ ์์ต๋๋ค. ๋คํธ์ํฌ์์ ๋๋ฐ์ด์ค์ ์ฐ๊ฒฐ์ ๋์ด๋ณด๋ฉด ์ปดํฌ๋ํธ๊ฐ ์๋ต์ผ๋ก ๋ฆฌ๋ ๋๋ง๋๋ ๊ฒ์ ํ์ธํ ์ ์์ต๋๋ค.
import { useSyncExternalStore } from 'react'; export default function ChatIndicator() { const isOnline = useSyncExternalStore(subscribe, getSnapshot); return <h1>{isOnline ? 'โ Online' : 'โ Disconnected'}</h1>; } function getSnapshot() { return navigator.onLine; } function subscribe(callback) { window.addEventListener('online', callback); window.addEventListener('offline', callback); return () => { window.removeEventListener('online', callback); window.removeEventListener('offline', callback); }; }
custom Hook์ผ๋ก ๋ก์ง ์ถ์ถํ๊ธฐ
์ผ๋ฐ์ ์ผ๋ก ์ปดํฌ๋ํธ์์ ์ง์ useSyncExternalStore๋ฅผ ์์ฑํ์ง๋ ์์ต๋๋ค. ๋์ ์ผ๋ฐ์ ์ผ๋ก custom Hook์์ ํธ์ถํฉ๋๋ค. ์ด๋ ๊ฒ ํ๋ฉด ์๋ก ๋ค๋ฅธ ์ปดํฌ๋ํธ์์ ๋์ผํ ์ธ๋ถ ์ ์ฅ์๋ฅผ ์ฌ์ฉํ ์ ์์ต๋๋ค.
์๋ฅผ ๋ค์ด ์ด custom useOnlineStatus Hook์ ๋คํธ์ํฌ๊ฐ ์จ๋ผ์ธ ์ํ์ธ์ง ์ฌ๋ถ๋ฅผ ์ถ์ ํฉ๋๋ค.
import { useSyncExternalStore } from 'react';
export function useOnlineStatus() {
const isOnline = useSyncExternalStore(subscribe, getSnapshot);
return isOnline;
}
function getSnapshot() {
// ...
}
function subscribe(callback) {
// ...
}์ด์ ๋ค๋ฅธ ์ปดํฌ๋ํธ์์ ๊ธฐ๋ณธ ๊ตฌํ์ ๋ฐ๋ณตํ์ง ์๊ณ ๋ useOnlineStatus๋ฅผ ํธ์ถํ ์ ์์ต๋๋ค.
import { useOnlineStatus } from './useOnlineStatus.js'; function StatusBar() { const isOnline = useOnlineStatus(); return <h1>{isOnline ? 'โ Online' : 'โ Disconnected'}</h1>; } function SaveButton() { const isOnline = useOnlineStatus(); function handleSaveClick() { console.log('โ Progress saved'); } return ( <button disabled={!isOnline} onClick={handleSaveClick}> {isOnline ? 'Save progress' : 'Reconnecting...'} </button> ); } export default function App() { return ( <> <SaveButton /> <StatusBar /> </> ); }
์๋ฒ ๋ ๋๋ง ์ง์ ์ถ๊ฐ
React ์ฑ์ด server rendering์ ์ฌ์ฉํ๋ ๊ฒฝ์ฐ React ์ปดํฌ๋ํธ๋ ๋ธ๋ผ์ฐ์ ํ๊ฒฝ ์ธ๋ถ์์๋ ์คํ๋์ด ์ด๊ธฐ HTML์ ์์ฑํฉ๋๋ค. ์ด๋ก ์ธํด ์ธ๋ถ store์ ์ฐ๊ฒฐํ ๋ ๋ช ๊ฐ์ง ๋ฌธ์ ๊ฐ ๋ฐ์ํฉ๋๋ค.
- ๋ธ๋ผ์ฐ์ ์ ์ฉ API์ ์ฐ๊ฒฐํ๋ ๊ฒฝ์ฐ ์๋ฒ์ ํด๋น API๊ฐ ์กด์ฌํ์ง ์์ผ๋ฏ๋ก ์๋ํ์ง ์์ต๋๋ค.
- ์๋ ํํฐ ๋ฐ์ดํฐ ์ ์ฅ์์ ์ฐ๊ฒฐํ๋ ๊ฒฝ์ฐ ์๋ฒ์ ํด๋ผ์ด์ธํธ ๊ฐ์ ์ผ์นํ๋ ๋ฐ์ดํฐ๊ฐ ํ์ํฉ๋๋ค.
์ด๋ฌํ ๋ฌธ์ ๋ฅผ ํด๊ฒฐํ๋ ค๋ฉด getServerSnapshot ํจ์๋ฅผ useSyncExternalStore์ ์ธ ๋ฒ์งธ ์ธ์๋ก ์ ๋ฌํ์ธ์.
import { useSyncExternalStore } from 'react';
export function useOnlineStatus() {
const isOnline = useSyncExternalStore(subscribe, getSnapshot, getServerSnapshot);
return isOnline;
}
function getSnapshot() {
return navigator.onLine;
}
function getServerSnapshot() {
return true; // ์๋ฒ์์ ์์ฑ๋ HTML์๋ ํญ์ "Online"์ ํ์ํฉ๋๋ค.
}
function subscribe(callback) {
// ...
}getServerSnapshot ํจ์๋ getSnapshot๊ณผ ์ ์ฌํ์ง๋ง ๋ ๊ฐ์ง ์ํฉ์์๋ง ์คํ๋ฉ๋๋ค.
- HTML์ ์์ฑํ ๋ ์๋ฒ์์ ์คํ๋ฉ๋๋ค.
- hydration ์ค ์ฆ React๊ฐ ์๋ฒ HTML์ ๊ฐ์ ธ์์ ์ธํฐ๋ํฐ๋ธํ๊ฒ ๋ง๋ค ๋ ํด๋ผ์ด์ธํธ์์ ์คํ๋ฉ๋๋ค.
์ด๋ฅผ ํตํด ์ฑ์ด ์ํธ์์ฉํ๊ธฐ ์ ์ ์ฌ์ฉ๋ ์ด๊ธฐ ์ค๋ ์ท ๊ฐ์ ์ ๊ณตํ ์ ์์ต๋๋ค. ์๋ฒ ๋ ๋๋ง์ ์๋ฏธ ์๋ ์ด๊ธฐ๊ฐ์ด ์๋ค๋ฉด ์ปดํฌ๋ํธ๊ฐ ํด๋ผ์ด์ธํธ์์๋ง ๋ ๋๋ง๋๋๋ก ๊ฐ์ ์ค์ ํ ์ ์์ต๋๋ค.
ํธ๋ฌ๋ธ ์ํ
์ค๋ฅ๊ฐ ๋ฐ์ํ์ต๋๋ค: โgetSnapshot์ ๊ฒฐ๊ณผ๋ฅผ ์บ์ํด์ผ ํฉ๋๋ค.โ
์ด ์ค๋ฅ๊ฐ ๋ฐ์ํ๋ฉด getSnapshot ํจ์๊ฐ ํธ์ถ๋ ๋๋ง๋ค ์ ๊ฐ์ฒด๋ฅผ ๋ฐํํ๋ค๋ ์๋ฏธ์
๋๋ค.
function getSnapshot() {
// ๐ด getSnapshot์์ ํญ์ ๋ค๋ฅธ ๊ฐ์ฒด๋ฅผ ๋ฐํํ์ง ๋ง์ธ์.
return {
todos: myStore.todos
};
}React๋ getSnapshot ๋ฐํ ๊ฐ์ด ์ง๋๋ฒ๊ณผ ๋ค๋ฅด๋ฉด ์ปดํฌ๋ํธ๋ฅผ ๋ฆฌ๋ ๋๋งํฉ๋๋ค. ๊ทธ๋ ๊ธฐ ๋๋ฌธ์ ํญ์ ๋ค๋ฅธ ๊ฐ์ ๋ฐํํ๋ฉด ๋ฌดํ ๋ฃจํ์ ๋ค์ด๊ฐ์ ์ด ์ค๋ฅ๊ฐ ๋ฐ์ํฉ๋๋ค.
์ค์ ๋ก ๋ณ๊ฒฝ๋ ์ฌํญ์ด ์๋ ๊ฒฝ์ฐ์๋ง getSnapshot ๊ฐ์ฒด๊ฐ ๋ค๋ฅธ ๊ฐ์ฒด๋ฅผ ๋ฐํํด์ผ ํฉ๋๋ค. store์ ๋ณ๊ฒฝ ๋ถ๊ฐ๋ฅํ ๋ฐ์ดํฐ๊ฐ ํฌํจ๋ ๊ฒฝ์ฐ ํด๋น ๋ฐ์ดํฐ๋ฅผ ์ง์ ๋ฐํํ ์ ์์ต๋๋ค.
function getSnapshot() {
// โ
๋ถ๋ณ ๋ฐ์ดํฐ๋ฅผ ๋ฐํํ ์ ์์ต๋๋ค.
return myStore.todos;
}store ๋ฐ์ดํฐ๊ฐ ๋ณ๊ฒฝ ๊ฐ๋ฅํ ๊ฒฝ์ฐ getSnapshot ํจ์๋ ํด๋น ๋ฐ์ดํฐ์ ๋ณ๊ฒฝ ๋ถ๊ฐ๋ฅํ ์ค๋
์ท์ ๋ฐํํด์ผ ํฉ๋๋ค. ์ฆ ์ ๊ฐ์ฒด๋ฅผ ์์ฑํด์ผ ํ์ง๋ง ๋งค๋ฒ ํธ์ถํ ๋๋ง๋ค ์ด ์์
์ ์ํํด์๋ ์ ๋ฉ๋๋ค. ๋์ ๋ง์ง๋ง์ผ๋ก ๊ณ์ฐ๋ ์ค๋
์ท์ ์ ์ฅํ๊ณ ์ ์ฅ์์ ๋ฐ์ดํฐ๊ฐ ๋ณ๊ฒฝ๋์ง ์์ ๊ฒฝ์ฐ ์ง๋๋ฒ๊ณผ ๋์ผํ ์ค๋
์ท์ ๋ฐํํด์ผ ํฉ๋๋ค. ๋ณ๊ฒฝ ๊ฐ๋ฅํ ๋ฐ์ดํฐ๊ฐ ๋ณ๊ฒฝ๋์๋์ง ํ์ธํ๋ ๋ฐฉ๋ฒ์ ๋ณ๊ฒฝ ๊ฐ๋ฅํ ์ ์ฅ์๊ฐ ๊ตฌํ๋ ๋ฐฉ์์ ๋ฐ๋ผ ๋ค๋ฆ
๋๋ค.
๋ฆฌ๋ ๋๋งํ ๋๋ง๋ค subscribe ํจ์๊ฐ ํธ์ถ๋ฉ๋๋ค.
subscribe ํจ์๋ ์ปดํฌ๋ํธ ๋ด๋ถ์ ์ ์๋๋ฏ๋ก ๋ฆฌ๋ ๋๋งํ ๋๋ง๋ค ๋ฌ๋ผ์ง๋๋ค.
function ChatIndicator() {
// ๐ฉ ํญ์ ๋ค๋ฅธ ํจ์๋ฅผ ์ฌ์ฉํ๋ฏ๋ก React๋ ๋ ๋๋งํ ๋๋ง๋ค ๋ค์ ๊ตฌ๋
ํฉ๋๋ค.
function subscribe() {
// ...
}
const isOnline = useSyncExternalStore(subscribe, getSnapshot);
// ...
}๋ฆฌ๋ ๋๋ง ์ฌ์ด์ ๋ค๋ฅธ subscribe ํจ์๋ฅผ ์ ๋ฌํ๋ฉด React๊ฐ store๋ฅผ ๋ค์ ๊ตฌ๋
ํฉ๋๋ค. ์ด๋ก ์ธํด ์ฑ๋ฅ ๋ฌธ์ ๊ฐ ๋ฐ์ํ๊ณ store ์ฌ๊ตฌ๋
์ ํผํ๊ณ ์ถ๋ค๋ฉด subscribe ํจ์๋ฅผ ์ธ๋ถ๋ก ์ด๋ํ์ธ์.
// โ
ํญ์ ๋์ผํ ํจ์์ด๋ฏ๋ก React๋ ๋ค์ ๊ตฌ๋
ํ ํ์๊ฐ ์์ต๋๋ค.
function subscribe() {
// ...
}
function ChatIndicator() {
const isOnline = useSyncExternalStore(subscribe, getSnapshot);
// ...
}๋๋ ์ผ๋ถ ์ธ์๊ฐ ๋ณ๊ฒฝ๋ ๋๋ง ๋ค์ ๊ตฌ๋
ํ๋๋ก subscribe์ useCallback์ผ๋ก ๋ํํฉ๋๋ค.
function ChatIndicator({ userId }) {
// โ
userId๊ฐ ๋ณ๊ฒฝ๋์ง ์๋ ํ ๋์ผํ ํจ์์
๋๋ค.
const subscribe = useCallback(() => {
// ...
}, [userId]);
const isOnline = useSyncExternalStore(subscribe, getSnapshot);
// ...
}