상세 컨텐츠

본문 제목

[리액트를 다루는 기술] 12장 immer를 사용하여 더 쉽게 불변성 유지하기

21-22/21-22 리액트 스타터 -1

by Kimpeep 2022. 1. 17. 13:00

본문

728x90

12. immer 설치 및 사용법

전개 연산자를 여러 번 사용하는 것은 꽤 번거롭고 가독성 또한 좋지 않습니다. immer 라이브러리를 사용하면, 구조가 복잡한 객체도 매우 쉽고 짧은 코드를 사용하여 불변성을 유지하면서 업데이트해 줄 수 있습니다.


12.1 immer 설치

$ yarn add immer

 

12.2 immer 사용하지 않고 불변성 유지

immer를 사용하지 않고 불변성을 유지하면서 값을 업데이트하는 컴포넌트

// App.js
import React, {useRef, useCallback, useState } from 'react';

const App = () => {
  const nextId = useRef(1);
  const [form, setForm] = useState({name: '', username: ''});
  const [data, setData] = useState({
    array: [],
    uselessValue: null
  });
    
  // input 수정을 위한 함수
  const onChange = useCallback(
    e => {
      const {name, value} = e.target;
      setForm({
        ...form,
        [name]: [value]
      });
    },
    [form]
  );

  // form 등록을 위한 함수
  const onSubmit = useCallback(
    e => {
      e.preventDefault();
      const info = {
        id: nextId.current,
        name: form.name,
        username: form.username
      };
          
      // array에 새 항목 등록
      setData({
        ...data,
        array: data.array.concat(info)
      });
            
      // form 초기화
      setForm({
         name: '',
         username: ''
      });
      nextId.current += 1;
    },
    [data, form.name, form.username]
  );
    
  // 항목을 삭제하는 함수
  const onRemove = useCallback(
    id => {
      setData({
        ...data,
        array: data.array.filter(info => info.id !== id)
      });
    },
    [data]
  );

  return (
    <div>
      <form onSubmit={onSubmit}>
        <input
           name="username"
           placeholder="아이디"
           value={form.username}
           onChange={onChange}
        />
        <input
           name="name"
           placeholder="이름"
           value={form.name}
           onChange={onChange} 
        />
		<button type="submit">등록</button>
      </form>
      <div>
        <ul>
          {data.array.map(info => (
            <li key={info.id} onClick={() => onRemove(info.id)}>
              {info.username} ({info.name})
            </li>
           ))}
        </ul>
      </div>
    </div>
  );
};

export default App;

전개 연산자와 배열 내장 함수를 사용하여 불변성을 유지하는 것은 어렵지 않지만, 상태가 복잡해진다면 귀찮은 작업이 될 수도 있습니다.

 

12.3 immer 사용법

immer를 사용하면 불변성을 유지하는 작업을 매우 간단하게 할 수 있습니다.

import produce from 'immer';
const nextState = produce(originalState, draft => {
  // 바꾸고 싶은 값 바꾸기
  draft.somewhere.deep.inside = 5;

produce 함수의 첫 번째 파라미터는 수정하고 싶은 상태, 두 번째 파라미터는 상태를 어떻게 업데이트할지 정의하는 함수입니다. 두 번째 파라미터로 전달되는 함수 내부에서 원하는 값을 변경하면 produce 함수가 불변성 유지를 대신해 주면서 새로운 상태를 생성합니다. 이 라이브러리의 핵심은 '불변성에 신경 쓰지 않는 것처럼 코드를 작성하되 불변성 관리는 제대로 해주는 것'입니다. 깊은 곳에 위치하는 값을 바꾸는 것 외에 배열을 처리할 때도 쉽고 편리합니다.

 

- 좀 더 복잡한 데이터를 불변성을 유지하면서 업데이트하는 예시

import produce from 'immer';

const originalState = [
  {
    id: 1,
    todo: '전개 연산자와 배열 내장 함수로 불변성 유지하기',
    checked: true,
  },
  {
    id: 2,
    todo: 'immer로 불변성 유지하기',
    checked: false,
  }
];

const nextState = produce(originalState, draft => {
  // id가 2인 항목의 checked 값을 true로 설정
  const todo = draft.find(t => t.id === 2); // id로 항목 찾기
  todo.checked = true;
    // 혹은 draft[1].checked = true;
    
  // 배열에 새로운 데이터 추가
  draft.push({
    id: 3,
    todo: '일정 관리 앱에 immer 적용하기',
    checked false,
  });
  
  // id = 1인 항목 제거하기
  draft.splice(draft.findIndex(t => t.id === 1), 1);
});

 

12.4 App 컴포넌트에 immer 적용하기

// App.js
import React, {useRef, useCallback, useState } from 'react';
import produce from 'immer';

(...)
    
  // input 수정을 위한 함수
  const onChange = useCallback(
    e => {
      const {name, value} = e.target;
      setForm(
		produce(form, draft => {
          draft[name] = value;
        })
      );
    },
    [form]
  );

  // form 등록을 위한 함수
  const onSubmit = useCallback(
    e => {
      e.preventDefault();
      const info = {
        id: nextId.current,
        name: form.name,
        username: form.username
      };
          
      // array에 새 항목 등록
      setData(
        produce(data, draft => {
          draft.array.push(info);
        })
      );
            
      // form 초기화
      setForm({
         name: '',
         username: ''
      });
      nextId.current += 1;
    },
    [data, form.name, form.username]
  );
    
  // 항목을 삭제하는 함수
  const onRemove = useCallback(
    id => {
      setData(
        produce(data, draft => {
          draft.array.splice(draft.array.findIndex(info => info.id === id), 1);
        })
      );
    },
    [data]
  );

 (...)

immer를 사용하여 컴포넌트 상태를 작성할 때는 객체 안의 값을 직접 수정하거나, 배열에 직접적인 변화를 일으키는 push, splice 등의 함수를 사용해도 됩니다. immer를 사용한다고 해서 무조건 코드가 간결해지지는 않습니다. onRemove의 경우, 배열 내장 함수 filter를 사용하는 것이 더 깔끔하므로 굳이 immer를 사용할 필요가 없습니다. immer는 불변성을 유지하는 코드가 복잡할 때만 사용해도 충분합니다.

 

12.5 useState의 함수형 업데이트와 immer 함께 쓰기

immer에서 제공하는 produce 함수를 호출할 때, 첫 번째 파라미터가 함수 형태라면 업데이트 함수를 반환합니다. 이러한 immer의 속성과 useState의 함수형 업데이트를 함께 사용하면 코드를 더욱 깔끔하게 작성할 수 있습니다.

// App.js
import React, {useRef, useCallback, useState } from 'react';
import produce from 'immer';

(...)
    
  // input 수정을 위한 함수
  const onChange = useCallback(
    e => {
      const {name, value} = e.target;
      setForm(
		produce(draft => {
          draft[name] = value;
        })
      );
    },
    [form]
  );

  // form 등록을 위한 함수
  const onSubmit = useCallback(
    e => {
      e.preventDefault();
      const info = {
        id: nextId.current,
        name: form.name,
        username: form.username
      };
          
      // array에 새 항목 등록
      setData(
        produce(draft => {
          draft.array.push(info);
        })
      );
            
      // form 초기화
      setForm({
         name: '',
         username: ''
      });
      nextId.current += 1;
    },
    [data, form.name, form.username]
  );
    
  // 항목을 삭제하는 함수
  const onRemove = useCallback(
    id => {
      setData(
        produce(draft => {
          draft.array.splice(draft.array.findIndex(info => info.id === id), 1);
        })
      );
    },
    [data]
  );

 (...)

 

12장 Quiz

1. ○○ ○○○를 사용하면 기존에 가지고 있던 값을 유지하면서 원하는 값을 새로 지정할 수 있지만, 객체의 구조가 깊어지면 ○○○을 유지하면서 업데이트하기 어렵다.

2. ○○○○○ 라이브러리를 사용하면 구조가 복잡한 객체도 쉽고 짧은 코드를 이용하여 ○○○을 유지하면서 업데이트해 줄 수 있다.

3. produce 함수는 두 가지 파라미터를 받는데 첫 번째 파라미터는 ○○하고 싶은 ○○, 두 번째 파라미터는 상태를 어떻게 ○○○○할지 정의하는 ○○이다.

4. immer 속성과 (      ) 를 함께 활용하면 코드를 더욱 깔끔하게 만들 수 있다.

더보기

<퀴즈 정답>

1. 전개 연산자, 불변성

2. immer, 불변성

3. 수정, 상태, 업데이트, 함수

4. useState의 함수형 업데이트

 


Corner React Starter #1
Editor dori

728x90

관련글 더보기