state 로직을 reducer로 작성하기
state를 업데이트하는 로직이 여러 이벤트 핸들러에 분산된 컴포넌트는 양이 많아 부담스러울 수 있습니다. 이 경우 state를 업데이트하는 모든 로직을 컴포넌트 외부에서 reducer라고 하는 단일 함수로 통합할 수 있습니다.
학습 내용
- reducer 함수란 무엇인가
useState
에서useReducer
로 리펙토링 하는 방법- reducer를 언제 사용할 수 있는지
- reducer를 잘 작성하는 방법
reducer를 사용하여 state 로직 통합하기
As your components grow in complexity, it can get harder to see at a glance all the different ways in which a component’s state gets updated. For example, the TaskApp
component below holds an array of tasks
in state and uses three different event handlers to add, remove, and edit tasks:
import { useState } from 'react'; import AddTask from './AddTask.js'; import TaskList from './TaskList.js'; export default function TaskApp() { const [tasks, setTasks] = useState(initialTasks); function handleAddTask(text) { setTasks([...tasks, { id: nextId++, text: text, done: false }]); } function handleChangeTask(task) { setTasks(tasks.map(t => { if (t.id === task.id) { return task; } else { return t; } })); } function handleDeleteTask(taskId) { setTasks( tasks.filter(t => t.id !== taskId) ); } return ( <> <h1>Prague itinerary</h1> <AddTask onAddTask={handleAddTask} /> <TaskList tasks={tasks} onChangeTask={handleChangeTask} onDeleteTask={handleDeleteTask} /> </> ); } let nextId = 3; const initialTasks = [ { id: 0, text: 'Visit Kafka Museum', done: true }, { id: 1, text: 'Watch a puppet show', done: false }, { id: 2, text: 'Lennon Wall pic', done: false }, ];
각 이벤트 핸들러는 state를 업데이트하기 위해 setTasks
를 호출합니다. 컴포넌트가 커질수록 그 안에서 state를 다루는 로직의 양도 늘어나게 됩니다. 복잡성를 줄이고 접근성을 높이기 위해서, 컴포넌트 내부에 있는 state 로직을 컴포넌트 외부의 “reducer”라고 하는 단일 함수로 옮길 수 있습니다.
reducer는 state를 다루는 다른 방법입니다. 다음과 같은 세가지 단계에 걸쳐 useState
에서 useReducer
로 바꿀 수 있습니다.
- state를 설정하는 것에서 action을 dispatch 함수로 전달하는 것으로 바꾸기.
- reducer 함수 작성하기.
- 컴포넌트에서 reducer 사용하기.
1단계: state를 설정하는 것에서 action을 dispatch 함수로 전달하는 것으로 바꾸기
현재 이벤트 핸들러는 state를 설정함으로써 무엇을 할 것인지를 명시합니다.
function handleAddTask(text) {
setTasks([...tasks, {
id: nextId++,
text: text,
done: false
}]);
}
function handleChangeTask(task) {
setTasks(tasks.map(t => {
if (t.id === task.id) {
return task;
} else {
return t;
}
}));
}
function handleDeleteTask(taskId) {
setTasks(
tasks.filter(t => t.id !== taskId)
);
}
위 코드에서 state 설정과 관련한 로직을 전부 지워보세요. 다음과 같이 세가지 이벤트 핸들러가 남게 될 것입니다.
- 사용자가 “Add” 를 눌렀을 때 호출되는
handleAddTask(text)
- 사용자가 task를 토글하거나 “저장”을 누르면 호출되는
handleChangeTask(task)
- 사용자가 “Delete” 를 누르면 호출되는
handleDeleteTask(taskId)
Managing state with reducers is slightly different from directly setting state. Instead of telling React “what to do” by setting state, you specify “what the user just did” by dispatching “actions” from your event handlers. (The state update logic will live elsewhere!) So instead of “setting tasks
” via an event handler, you’re dispatching an “added/changed/deleted a task” action. This is more descriptive of the user’s intent.
function handleAddTask(text) {
dispatch({
type: 'added',
id: nextId++,
text: text,
});
}
function handleChangeTask(task) {
dispatch({
type: 'changed',
task: task
});
}
function handleDeleteTask(taskId) {
dispatch({
type: 'deleted',
id: taskId
});
}
dispatch
함수에 넣어준 객체를 “action” 이라고 합니다.
function handleDeleteTask(taskId) {
dispatch(
// "action" 객체:
{
type: 'deleted',
id: taskId
}
);
}
이 객체는 일반적인 자바스크립트 객체입니다. 이 안에 어떤 것이든 자유롭게 넣을 수 있지만, 일반적으로 어떤 상황이 발생하는지에 대한 최소한의 정보를 담고 있어야합니다. (dispatch
함수 자체에 대한 부분은 이후 단계에서 다룰 예정입니다.)
2단계: reducer 함수 작성하기
reducer 함수는 state에 대한 로직을 넣는 곳 입니다. 이 함수는 현재의 state 값과 action 객체, 이렇게 두개의 인자를 받고 다음의 state 값을 반환해줍니다.
function yourReducer(state, action) {
// React가 설정하게될 다음 state 값을 반환합니다.
}
React는 reducer에서 반환한 값을 state에 설정합니다.
이 예시에서 이벤트 핸들러에 구현 되어있는 state 설정과 관련한 로직을 reducer 함수로 옮기기 위해서, 다음과 같이 해볼 것입니다.
- 첫 번째 인자에 현재 state (
tasks
) 선언하기. - 두 번째 인자에
action
객체 선언하기. - reducer에서 다음 state 반환하기. (React가 state에 설정하게 될 값)
다음은 state 설정과 관련한 로직을 reducer 함수로 마이그레이션한 코드입니다.
function tasksReducer(tasks, action) {
if (action.type === 'added') {
return [...tasks, {
id: action.id,
text: action.text,
done: false
}];
} else if (action.type === 'changed') {
return tasks.map(t => {
if (t.id === action.task.id) {
return action.task;
} else {
return t;
}
});
} else if (action.type === 'deleted') {
return tasks.filter(t => t.id !== action.id);
} else {
throw Error('Unknown action: ' + action.type);
}
}
reducer 함수는 state(tasks
)를 인수로 받고 있기 때문에, 이를 컴포넌트 외부에서 선언할 수 있습니다. 이렇게 하면 들여쓰기 수준이 줄어들고 코드를 더 쉽게 읽을 수 있게 될 것입니다.
Deep Dive
reducer를 사용하면 컴포넌트 내부의 코드 양을 “줄일 수” 있지만, 실제로는 배열에서 사용할 수 있는 reduce()
연산의 이름을 따서 명명되었습니다.
reduce()
는 배열의 여러 값을 단일 값으로 “누적”하는 연산을 수행합니다.
const arr = [1, 2, 3, 4, 5];
const sum = arr.reduce(
(result, number) => result + number
); // 1 + 2 + 3 + 4 + 5
이때 reduce
에 전달하는 함수는 “reducer”로 알려져 있습니다. 이 함수는 _지금까지의 결과_와 _현재 아이템_을 인자로 받고 _다음 결과_를 반환합니다. 비슷한 아이디어의 예로 React의 reducer는 _지금까지의 state_와 _action_을 인자로 받고 _다음 state_를 반환합니다. 이 과정에서 여러 action을 누적하여 state로 반환합니다.
initialState
와 reducer 함수를 넘겨 받아 최종적인 state 값으로 계산하기 위한 action
배열을 인자로 받는 reduce()
메서드를 사용할 수도 있습니다.
import tasksReducer from './tasksReducer.js'; let initialState = []; let actions = [ { type: 'added', id: 1, text: 'Visit Kafka Museum' }, { type: 'added', id: 2, text: 'Watch a puppet show' }, { type: 'deleted', id: 1 }, { type: 'added', id: 3, text: 'Lennon Wall pic' }, ]; let finalState = actions.reduce( tasksReducer, initialState ); const output = document.getElementById('output'); output.textContent = JSON.stringify( finalState, null, 2 );
여러분이 직접 구현 할 일은 거의 없지만 위 예시는 React에 구현되어 있는 것과 비슷합니다.
3단계: 컴포넌트에서 reducer 사용하기
마지막으로 tasksReducer
를 컴포넌트에 연결할 차례입니다. React에서 useReducer
Hook을 불러와주세요.
import { useReducer } from 'react';
그런 다음, useState
를
const [tasks, setTasks] = useState(initialTasks);
아래 처럼 useReducer
로 바꿔주세요.
const [tasks, dispatch] = useReducer(tasksReducer, initialTasks);
useReducer
Hook은 초기 state 값을 입력받고 state를 담을 수 있는 값과 state를 설정하는 함수(useReducer의 경우는 dispatch 함수를 의미)를 반환하는 점으로 보면 useState
와 비슷합니다. 하지만 조금 다른 점이 있습니다.
useReducer
Hook은 두개의 인자를 넘겨 받습니다.
- reducer 함수
- 초기 state 값
그리고 아래와 같이 반환합니다.
- state를 담을 수 있는 값
- dispatch 함수 (사용자의 action을 reducer 함수에게 “전달하게 될”)
이제 준비가 다 되었습니다! 아래 예시의 컴포넌트 파일 아래에는 reducer가 선언되어있습니다.
import { useReducer } from 'react'; import AddTask from './AddTask.js'; import TaskList from './TaskList.js'; export default function TaskApp() { const [tasks, dispatch] = useReducer( tasksReducer, initialTasks ); function handleAddTask(text) { dispatch({ type: 'added', id: nextId++, text: text, }); } function handleChangeTask(task) { dispatch({ type: 'changed', task: task }); } function handleDeleteTask(taskId) { dispatch({ type: 'deleted', id: taskId }); } return ( <> <h1>Prague itinerary</h1> <AddTask onAddTask={handleAddTask} /> <TaskList tasks={tasks} onChangeTask={handleChangeTask} onDeleteTask={handleDeleteTask} /> </> ); } function tasksReducer(tasks, action) { switch (action.type) { case 'added': { return [...tasks, { id: action.id, text: action.text, done: false }]; } case 'changed': { return tasks.map(t => { if (t.id === action.task.id) { return action.task; } else { return t; } }); } case 'deleted': { return tasks.filter(t => t.id !== action.id); } default: { throw Error('Unknown action: ' + action.type); } } } let nextId = 3; const initialTasks = [ { id: 0, text: 'Visit Kafka Museum', done: true }, { id: 1, text: 'Watch a puppet show', done: false }, { id: 2, text: 'Lennon Wall pic', done: false } ];
아래 처럼 reducer를 다른 파일로 분리하는 것도 가능합니다.
import { useReducer } from 'react'; import AddTask from './AddTask.js'; import TaskList from './TaskList.js'; import tasksReducer from './tasksReducer.js'; export default function TaskApp() { const [tasks, dispatch] = useReducer( tasksReducer, initialTasks ); function handleAddTask(text) { dispatch({ type: 'added', id: nextId++, text: text, }); } function handleChangeTask(task) { dispatch({ type: 'changed', task: task }); } function handleDeleteTask(taskId) { dispatch({ type: 'deleted', id: taskId }); } return ( <> <h1>Prague itinerary</h1> <AddTask onAddTask={handleAddTask} /> <TaskList tasks={tasks} onChangeTask={handleChangeTask} onDeleteTask={handleDeleteTask} /> </> ); } let nextId = 3; const initialTasks = [ { id: 0, text: 'Visit Kafka Museum', done: true }, { id: 1, text: 'Watch a puppet show', done: false }, { id: 2, text: 'Lennon Wall pic', done: false }, ];
관심사를 분리하면 컴포넌트의 로직은 읽기 더 쉬워질 수 있습니다. 이렇게 하면 이벤트 핸들러는 action을 전달해줘서 무슨 일이 일어났는지에 관련한 것만 명시하면 되고 reducer 함수는 이에 대한 응답으로 state가 어떤 값으로 업데이트 될지를 결정하기만 하면 됩니다.
useState
와 useReducer
비교하기
reducer가 좋은 점만 있는 것은 아닙니다! 아래에서 useState
와 useReducer
를 비교할 수 있는 몇 가지 방법을 소개하겠습니다.
- 코드 크기: 일반적으로
useState
를 사용하면, 미리 작성해야 하는 코드가 줄어듭니다.useReducer
를 사용하면 reducer 함수 그리고 action을 전달하는 부분 둘 다 작성해야 합니다. 그렇지만 여러 이벤트 핸들러에서 비슷한 방식으로 state를 업데이트하는 경우,useReducer
를 사용하면 코드의 양을 줄이는 데 도움이 될 수 있습니다. - 가독성:
useState
로 간단한 state를 업데이트하는 경우 가독성이 좋은 편입니다. 그렇지만 더 복잡한 구조의 state를 다루게 되면 컴포넌트의 코드 양이 더 많아져 한눈에 읽기 어려워질 수 있습니다. 이 경우useReducer
를 사용하면, 업데이트 로직이 어떻게 동작하는지와 이벤트 핸들러를 통해서 무엇이 발생했는지 구현한 부분을 명확하게 구분할 수 있습니다. - 디버깅:
useState
를 사용하며 버그를 발견했을 때, 왜, 어디서 state가 잘못 설정됐는지 찾기 어려울 수 있습니다.useReducer
를 사용하면, 콘솔 로그를 reducer에 추가하여 state가 업데이트되는 모든 부분과 왜 해당 버그가 발생했는지(어떤action
으로 인한 것인지)를 확인할 수 있습니다. 각action
이 올바르게 작성되어 있다면, 버그를 발생시킨 부분이 reducer 로직 자체에 있다는 것을 알 수 있을 것입니다. 그렇지만useState
를 사용하는 경우보다 더 많은 코드를 단계별로 실행해서 디버깅 해야 하는 점이 있기도 합니다. - 테스팅: reducer는 컴포넌트에 의존하지 않는 순수 함수입니다. 이는 reducer를 독립적으로 분리해서 내보내거나 테스트할 수 있다는 것을 의미합니다. 일반적으로 더 현실적인 환경에서 컴포넌트를 테스트하는 것이 좋지만, 복잡한 state를 업데이트하는 로직의 경우 reducer가 특정 초기 state 및 action에 대해 특정 state를 반환한다고 생각하고 테스트하는 것이 유용할 수 있습니다.
- 개인적인 취향: reducer를 좋아하는 사람도 있지만, 그렇지 않는 사람들도 있습니다. 괜찮습니다. 이건 선호도의 문제이니까요.
useState
와useReducer
는 동일한 방식이기 때문에 언제나 마음대로 바꿔서 사용해도 무방합니다.
만약 일부 컴포넌트에서 잘못된 방식으로 state를 업데이트하는 것으로 인한 버그가 자주 발생하거나 해당 코드에 더 많은 구조를 도입하고 싶다면 reducer 사용을 권장합니다. 이때 모든 부분에 reducer를 적용하지 않아도 됩니다. useState
and useReducer
를 마음대로 섞고 매치하세요! 이 둘은 심지어 같은 컴포넌트 안에서 사용해도 됩니다.
reducer 잘 작성하기
reducer를 작성할 때 다음과 같은 두가지 팁을 명심하세요.
- Reducers must be pure. Similar to state updater functions, reducers run during rendering! (Actions are queued until the next render.) This means that reducers must be pure—same inputs always result in the same output. They should not send requests, schedule timeouts, or perform any side effects (operations that impact things outside the component). They should update objects and arrays without mutations.
- Each action describes a single user interaction, even if that leads to multiple changes in the data. For example, if a user presses “Reset” on a form with five fields managed by a reducer, it makes more sense to dispatch one
reset_form
action rather than five separateset_field
actions. If you log every action in a reducer, that log should be clear enough for you to reconstruct what interactions or responses happened in what order. This helps with debugging!
Immer로 간결한 reducer 작성하기
일반적인 state에서 객체와 배열을 업데이트 하는 것처럼, Immer 라이브러리를 사용하면 reducer를 더 간결하게 작성할 수 있습니다. 이 라이브러리에서 제공하는 useImmerReducer
를 사용하여 push
또는 arr[i] =
로 값을 할당하므로써 state를 변경해보겠습니다.
import { useImmerReducer } from 'use-immer'; import AddTask from './AddTask.js'; import TaskList from './TaskList.js'; function tasksReducer(draft, action) { switch (action.type) { case 'added': { draft.push({ id: action.id, text: action.text, done: false }); break; } case 'changed': { const index = draft.findIndex(t => t.id === action.task.id ); draft[index] = action.task; break; } case 'deleted': { return draft.filter(t => t.id !== action.id); } default: { throw Error('Unknown action: ' + action.type); } } } export default function TaskApp() { const [tasks, dispatch] = useImmerReducer( tasksReducer, initialTasks ); function handleAddTask(text) { dispatch({ type: 'added', id: nextId++, text: text, }); } function handleChangeTask(task) { dispatch({ type: 'changed', task: task }); } function handleDeleteTask(taskId) { dispatch({ type: 'deleted', id: taskId }); } return ( <> <h1>Prague itinerary</h1> <AddTask onAddTask={handleAddTask} /> <TaskList tasks={tasks} onChangeTask={handleChangeTask} onDeleteTask={handleDeleteTask} /> </> ); } let nextId = 3; const initialTasks = [ { id: 0, text: 'Visit Kafka Museum', done: true }, { id: 1, text: 'Watch a puppet show', done: false }, { id: 2, text: 'Lennon Wall pic', done: false }, ];
reducer는 순수해야 하기 때문에, 이 안에서는 state를 변형할 수 없습니다. 그러나, Immer에서 제공하는 특별한 draft
객체를 사용하면 안전하게 state를 변형할 수 있습니다. 내부적으로, Immer는 변경 사항이 반영된 초안(draft)
으로 state의 복사본을 생성합니다. 이것이 useImmerReducer
가 관리하는 reducer가 첫 번째 인수인 state를 변형할 수 있고 새로운 state 값을 반환할 필요가 없는 이유입니다.
요약
- To convert from
useState
touseReducer
:- Dispatch actions from event handlers.
- Write a reducer function that returns the next state for a given state and action.
- Replace
useState
withuseReducer
.
- Reducers require you to write a bit more code, but they help with debugging and testing.
- Reducers must be pure.
- Each action describes a single user interaction.
- Use Immer if you want to write reducers in a mutating style.
챌린지 1 of 4: 이벤트 핸들러에서 action 전달하기
현재 ContactList.js
와 Chat.js
의 이벤트 핸들러 안에는 // TODO
주석이 있습니다. 이 때문에 input에 값을 입력해도 동작하지 않고 탭 버튼을 클릭해도 선택된 수신인이 변경되지 않습니다.
// TODO
주석이 있는 부분을 지우고 상황에 맞는 action을 전달(dispatch)
하는 코드를 작성해보세요. action에 대한 힌트를 얻고 싶다면 messengerReducer.js
에 구현된 reducer를 확인해보세요. 이 reducer는 이미 작성되어있기 때문에 변경할 필요가 없습니다. 여러분은 ContactList.js
와 Chat.js
에 action을 담아 전달하는 코드를 작성하기만 하면 됩니다.
import { useReducer } from 'react'; import Chat from './Chat.js'; import ContactList from './ContactList.js'; import { initialState, messengerReducer } from './messengerReducer'; export default function Messenger() { const [state, dispatch] = useReducer( messengerReducer, initialState ); const message = state.message; const contact = contacts.find(c => c.id === state.selectedId ); return ( <div> <ContactList contacts={contacts} selectedId={state.selectedId} dispatch={dispatch} /> <Chat key={contact.id} message={message} contact={contact} dispatch={dispatch} /> </div> ); } const contacts = [ { id: 0, name: 'Taylor', email: 'taylor@mail.com' }, { id: 1, name: 'Alice', email: 'alice@mail.com' }, { id: 2, name: 'Bob', email: 'bob@mail.com' } ];