Tech

Diary

Lecture

About Me

개발중

Context API 최적화

JeongSeulho

2023년 08월 20일

준비중...
클립보드로 복사
thumbnail

0. 들어가며

공통 프로젝트에서 Context API를 사용하였다, 프로젝트가 끝나고 코드를 리뷰하던 중 Context API에서 불필요한 렌더링이 발생하는 것을 발견하였다.
이러한 문제가 발생하는 이유와 해결 방법을 정리해 보았다.

1. Context API의 불필요한 렌더링

다음과 같이 Context API를 사용하여 상태를 관리하였다.

copy
export function UserInfoProvider({ children }: LayoutChildrenProps) {
  const [email, setEmail] = useState<string>("");
  const [nickname, setNickname] = useState<string>("");
  const value = { email, setEmail, nickname, setNickname }

  return <UserInfoContext.Provider value={value}>{children}</UserInfoContext.Provider>;
}

export function useUserInfoState() {
  const context = useContext(UserInfoContext);
  if (!context) {
    throw new Error("Cannot find Provider");
  }
  return context;
}

email, setEmail, nickname, setNickname를 객체로 묶어서 Contextvalue로 사용하였다. 각 컴포넌트에서 다음과 같이 Context를 사용한다고 해보자

copy
function UsingEmailComponent() {
  const { email } = useUserInfoState();
  return <div>{email}</div>;
}
copy
function UsingSetEmailComponent() {
  const { setEmail } = useUserInfoState();
  return <div onClick={() => setEmail("someEmail")}>setEmail</div>;
}
copy
function UsingNicknameComponent() {
  const { nickname } = useUserInfoState();
  return <div>{nickname}</div>;
}
copy
function UsingSetNicknameComponent() {
  const { setNickname } = useUserInfoState();
  return <div onClick={() => setNickname("someNickname")}>setNickname</div>;
}
copy
function App() {
  return (
    <UserInfoProvider>
      <UsingEmailComponent />
      <UsingSetEmailComponent />
      <UsingNicknameComponent />
      <UsingSetNicknameComponent />
    </UserInfoProvider>
  );
}

여기서 만약 setEmail을 호출하여 email 상태를 바꾼다면 해당 Context를 사용하는 모든 컴포넌트가 리렌더링된다.

  1. setEmail을 호출하여 email 상태를 바꾼다.
  2. email 상태를 사용하는 UserInfoProvider가 리렌더링된다.
  3. UserInfoProvider가 다시 실행되면서 value가 새로 생성된다(새로운 참조 값 생성).
  4. value를 사용하는 모든 컴포넌트가 리렌더링된다.

이것이 Context API의 문제인데, email만 바뀌어도 nickname을 사용하는 컴포넌트와, setEmail, setNickname을 사용하는 컴포넌트가 모두 리렌더링되는 것이다.

2. Context API 최적화

이러한 문제는 value라는 한 객체에 모든 상태와 함수를 넣어두어 서로 영향을 주기 때문에 발생한다.
이러한 문제를 해결하기 위해서 value를 분리하여 해결할 수 있다.

(1) Context 관심사 분리

emailnickname을 분리하여 각각의 Context를 만들면 된다.

copy
export function UserEmailProvider({ children }: LayoutChildrenProps) {
  const [email, setEmail] = useState<string>("");
  const value = { email, setEmail }

  return <UserEmailContext.Provider value={value}>{children}</UserEmailContext.Provider>;
}
copy
export function UserNicknameProvider({ children }: LayoutChildrenProps) {
  const [nickname, setNickname] = useState<string>("");
  const value = { nickname, setNickname }

  return <UserNicknameContext.Provider value={value}>{children}</UserNicknameContext.Provider>;
}
copy
function App() {
  return (
    <UserEmailProvider>
      <UserNicknameProvider>

        ...

      </UserNicknameProvider>
    </UserEmailProvider>
  );
}

이렇게 하면 setNickname을 호출에 email과 관련된 컴포넌트는 리렌더링되지 않도록 할 수 있다.

하지만 이 경우에도 setNickname을 호출 시

  1. UserNicknameProvider가 리렌더링된다.
  2. UserNicknameProvider가 다시 실행되면서 value가 새로 생성된다(새로운 참조 값 생성).
  3. setNickname을 사용만 하는 컴포넌트도 리렌더링된다.

일반적으로 nickname을 사용하는 컴포넌트만 리렌더링 되도록 하는 것이 올바를 것이다.
이 문제 또한, 값과 함수를 분리하며 함수를 useMemo로 감싸주어 새로운 참조값이 생성되지 않도록 하면 해결할 수 있다.

(2) 값과 함수를 분리, useMemo 사용

setNickname을 호출하여도 setNickname을 사용만하는 컴포넌트는 리렌더링 하지 않도록 해보자.

copy
export function UserNicknameProvider({ children }: LayoutChildrenProps) {
  const [nickname, setNickname] = useState<string>("");
  const action = useMemo(() => ({ setNickname }), [setNickname]);

  return (
    <UserNicknameActionContext.Provider value={action}>
      <UserNicknameStateContext.Provider value={{nickname}}>
        {children}
      </UserNicknameStateContext.Provider>
    </UserNicknameActionContext.Provider>
  )
}

이렇게 값과 함수를 분리하고, useMemo를 사용하여 Provider가 리렌더링 되어도 setNickname이 새로 생성되지 않도록 하였다.

3. 드러나는 Context API의 단점

(1) 길어지는 코드

관심사마다 ContextProvider를 만들고 해당 관심사안에서도 값과 함수를 분리하여야 한다. 또한, 그 많아진 ProviderApp에 넣어주어야 한다.

copy
function App() {
  return (
    <AProvider>
      <BProvider>
        <CProvider>
          <DProvider>

        ...
          
            </DProvider>
          </CProvider>
      </BProvider>
    </AProvider>
  );
}

(2) useMemo는 공짜가 아니다.

Context API를 리렌더링 방지를 위해선 useMemo를 사용하여야 한다.
useMemo를 사용한다는 것은 리렌더링의 비용 대신

  1. 메모리의 추가 사용
  2. CPU 연산의 추가 사용(의존성 배열의 값이 변경되었는지 확인)

위와 같은 비용을 지불하는 것이다.

4. 마치며

프로젝트 초기에는 Context API에대한 이해가 부족하였던 것 같다. 다음 프로젝트에서는 전역 상태를 위한 다른 방법도 고려해보고, Context API를 사용한다면 최적화에 대해 더 고민해봐야겠다.