-
Notifications
You must be signed in to change notification settings - Fork 1
React Suspense 적용
dannysir edited this page Dec 4, 2024
·
1 revision
Suspense는 React 16.6에서 도입된 기능으로, 컴포넌트의 렌더링을 특정 작업이 끝날 때까지 '중단'시키고 그동안 대체 UI를 보여줄 수 있게 해주는 기능입니다. 주로 데이터 로딩이나 코드 스플리팅 상황에서 사용됩니다.
기존에는 주로 다음과 같은 방식으로 로딩 상태를 처리했습니다
function ParentComponent() {
const { data, isLoading } = useQuery('data', fetchData);
return (
<div>
<ChildComponent isLoading={isLoading} data={data} />
<OtherComponent isLoading={isLoading} data={data} />
</div>
);
}
function ChildComponent({ isLoading, data }) {
if (isLoading) {
return <LoadingSpinner />;
}
return <div>{data}</div>;
}
이 방식의 문제점
- 로딩 상태를 props로 계속 전달해야 함
- 컴포넌트 간 결합도가 높아짐
- 세분화된 로딩 상태 관리가 어려움
Suspense는 이러한 문제를 해결하기 위해 다른 접근 방식을 사용합니다
function ParentComponent() {
return (
<Suspense fallback={<LoadingSpinner />}>
<ChildComponent />
<OtherComponent />
</Suspense>
);
}
function ChildComponent() {
const { data } = useQuery('data', fetchData, { suspense: true });
return <div>{data}</div>;
}
Suspense의 특징
- 로딩 상태를 상위 컴포넌트에서 선언적으로 처리
- 개별 컴포넌트는 로딩 상태를 신경 쓸 필요 없음
- 경계(boundary)를 기준으로 로딩 UI가 표시됨
다음은 주식 정보를 보여주는 홈 페이지의 구현 예시입니다
import { Suspense } from 'react';
import TopFive from 'components/TopFive';
import StockIndex from 'components/StockIndex';
import News from 'components/News';
// ... 스켈레톤 컴포넌트 import
export default function Home() {
return (
<>
<Suspense fallback={<StockIndexSkeleton />}>
<StockIndex />
</Suspense>
<Suspense fallback={<SkeletonList />}>
<TopFive />
</Suspense>
<Suspense fallback={<NewsSkeleton />}>
<News />
</Suspense>
</>
);
}
- Suspense는 항상 외부에서 컴포넌트를 감싸는 형태로 사용
- 개별 요소 단위의 로딩 처리가 필요한 경우에는 적합하지 않을 수 있음
- 컴포넌트를 적절한 크기로 분리하여 Suspense 경계 설정 필요
- props를 통한 isLoading 전달 방식에서 Suspense로의 전환 시 구조 변경 필요
- 점진적인 마이그레이션을 위해 두 방식의 혼용이 필요할 수 있음
- 컴포넌트 리팩토링 시 책임 범위 재정의 필요
React Query에서 Suspense를 사용하려면 다음과 같이 설정할 수 있습니다
const { data } = useQuery({
queryKey: ['stocks'],
queryFn: fetchStockData,
suspense: true // Suspense 모드 활성화
});
또는 QueryClient 설정에서 전역적으로 설정할 수도 있습니다
const queryClient = new QueryClient({
defaultOptions: {
queries: {
suspense: true
}
}
});
Suspense는 기존의 prop 기반 로딩 처리 방식과는 조금 다른 느낌이라고 생각합니다. 컴포넌트 단위의 세밀한 로딩 제어는 어려울 수 있지만, 선언적이고 깔끔한 코드 구조를 가능하게 합니다.
- [FE] 프론트엔드 기술스택
- [FE] 라이브러리 없이 차트 구현 이유
- [FE] Canvas API 사용방법
- [FE] 네비게이션 바 애니메이션 구현
- [FE] Socket.io 사용방법
- [FE] Tanstack Router에 대하여...
- [FE] Intl(Internationalization) API
- [FE] React Suspense 적용
- [FE] 한글 입력 방식의 유연성을 높인 검색 시스템 구현하기
- [BE] 백엔드 기술 스택
- [BE] SSE vs Socket.io
- [BE] Redis를 도입하게 된 계기
- [BE] ACG Rule을 활용한 Secure CI CD 파이프라인 구현
- [BE] Nginx 로드밸런싱을 통해 한국 투자 API 소켓 제한 극복
- [BE] 주가 지수 기능 개발 과정
- [BE] 매수 및 매도 기능 개발 과정
- [BE] 실시간 자산 조회 기능 개발 과정
- [BE] 단위 테스트
- [BE] redis를 이용한 한국투자 Open API 세션 관리
- [BE] 데이터베이스 인덱싱
- [FE] React에서의 DOM 요소 접근 (useRef vs getElementById)
- [FE] Outlet을 활용한 공통 레이아웃 관리
- [FE] react hooks가 특정 조건에서 실행되면 안되는 이유 & useQuery에 query function 매개변수가 undefined일 수도 있을 때 어떻게 해결할까
- [FE] cross‐domain 로컬 환경에서 cookie로 인증 처리하기 with vite proxy
- [FE] 크롬&사파리 Composition 차이
- [FE] useEffect 의존성 배열
- [BE] Naver Cloud Platform HTTPS 무응답 현상
- [BE] 한국투자 Open API에서 access token을 발급받지 못하는 문제
- [BE] 한국투자 Open API와 웹소켓 연결이 되지 않던 문제
- [BE] 한국투자 Open API 웹소켓 연결이 중단되는 문제
- [BE] 같은 주식 주문이 동시에 여러 번 체결되는 문제
- [BE] 한국투자 Open API Websocket 세션을 두 개에서 한 개로 변경하기
- [BE] Nginx 로드 밸런싱 중 Socket bad Request 발생하는 현상
- [BE] 매수/매도 체결 로직에 의해 redis pub/sub이 정상적으로 동작하지 않는 문제