
이번 글에서는 React에서 state management(상태 관리)를 할 때
널리 사용되어지고 있는 React Redux에 대해 알아보고자 한다.
⚠️⚠️⚠️⚠️⚠️⚠️⚠️⚠️⚠️⚠️⚠️⚠️⚠️⚠️⚠️⚠️⚠️⚠️⚠️
이 글은 Redux의 기초 개념을 어느정도 이해하고 있는 상태로 작성된 글입니다.
Store, Action, Reducer, Dispatch에 대해 처음 들어보시는 등
Redux에 대해 전혀 접근해보지 못한 분들은
제 블로그의 Redux 입문을 살펴보시는 것을 추천드립니다.
⚠️⚠️⚠️⚠️⚠️⚠️⚠️⚠️⚠️⚠️⚠️⚠️⚠️⚠️⚠️⚠️⚠️⚠️⚠️
React Redux is maintained by the Redux team, and kept up-to-date with the latest APIs from Redux and React.
Redux와 React Redux 모두, state management(상태 관리)를 위해 쓰이는 Redux라는 것은 다를 게 없으나
React Redux는 Redux 팀에서 공식적(Official)으로 관리, 배포하고 있는
React 전용 Redux 패키지이다.
그래서 Redux와 React Redux의 공식 문서는 각각 개별적으로 존재하고 있다.
Redux를 시작하기 전에 바로 React Redux를 사용해도 무방하나,
React Redux는 Redux를 기초로 설계되어있다.
그래서 시간이 좀 소요되더라도, 더 편하게 React Redux를 다루기 위해서는
Redux에 대한 기초 개념을 이해한 뒤에 React Redux를 사용하는 것이 편하다고 생각한다.
(↑ 개인적인 생각입니다. 이 학습 순서가 나에게는 더 적합했다.)
⚠️⚠️⚠️⚠️⚠️⚠️⚠️⚠️⚠️⚠️⚠️⚠️⚠️⚠️⚠️⚠️⚠️⚠️⚠️
그러므로 이 글은 Redux의 기초 개념을 어느정도 이해하고 있는 상태로 작성된 글입니다.
Store, Action, Reducer, Dispatch에 대해 처음 들어보시는 등
Redux에 대해 전혀 접근해보지 못한 분들은
제 블로그의 Redux 입문을 살펴보시는 것을 추천드립니다.
⚠️⚠️⚠️⚠️⚠️⚠️⚠️⚠️⚠️⚠️⚠️⚠️⚠️⚠️⚠️⚠️⚠️⚠️⚠️
React Redux에는 설치 방법이 2가지가 있다.
이 방법은 React 프로젝트 생성 시작부터 React Redux를 사용할 예정일 때 사용하면 유용하다.
React Redux별도로 설치하지 않아도 React Redux가 포함된 채로 프로젝트를 시작할 수 있기 때문이다.
npx create-react-app my-app --template redux
React 프로젝트를 생성할 당시에 React Redux를 함께 설치하지 못했고
이미 진행 중인 프로젝트에 뒤늦게 추가로만 React Redux를 추가할 때 사용.
npm i react-redux
또는
yarn add react-redux
이번 글에서는 To-do List를 React Redux를 이용하여 구현해보며,
React Redux의 기초 구문을 사용해보고자 한다.
이 코드에 React Redux를 도입하여 To-do List를 완성시켜나가고자 한다.
바탕이 될 코드는 createStore를 이용하여 간단하게 Store를 만들어뒀고,
reducer 함수도 작성해둔 상태이다.
Redux를 사용함에 있어서 제일 핵심이 되는 기술은 바로 Store이다.
React로 만들어진 애플리케이션은 대체로 다수의 컴포넌트로 이루어져있는데,
이 수많은 컴포넌트들이 Store에 access(접근)할 수 있게 하기 위해서는 어떻게 해야할까?
이 때 사용되는 것이 바로 Provider라는 컴포넌트이다.
- The Provider component makes the Redux store available to any nested components that need to access the Redux store.
- 출처: Provider - React Redux 공식 문서
Provider란 하나의 컴포넌트이며,
Provider 컴포넌트로 React 컴포넌트를 감싸줌으로써
Provider 컴포넌트 하위 컴포넌트들이 Provider를 통해 Redux의 store에 access(접근)할 수 있게 해준다.
이것은 예시를 보는 것이 더 편하다.
// index.js
import React from "react";
import ReactDOM from "react-dom";
import { Provider } from "react-redux";
import App from "./components/App";
import store from "./store";
ReactDOM.render(
<Provider store={store}>
<App />
</Provider>,
document.getElementById("root")
);
React Redux로부터 Provider 컴포넌트를 import 한 후,
Provider 컴포넌트로 store를 이용할 컴포넌트를 감싸준다.
그 후 Provider 컴포넌트의 props로 해당 store를 설정해주면 된다.
여기서 App 컴포넌트를 Provider 컴포넌트로 감싸준 이유는,
App 컴포넌트가 우리가 만든 프로젝트의 컴포넌트 중 가장 상위에 있는 컴포넌트이기 때문이다.
최상위 컴포넌트인 App 컴포넌트를 감싸준다면 그 하위의 모든 컴포넌트에서 당연히 store를 access(접근)할 수 있게 될 것이기 때문이다.
// store.js
const toDosStore = createStore(reducer);
export default toDosStore;
그리고 우리는 템플릿 코드와 ↑ 이 코드를 보다시피,
store.js라는 별도의 파일 내에서 createStore()를 이용하여 store를 생성해줬고,
그 store를 export해줬다.
export한 store를 index.js에서 import했으며
그 store를 Provider 컴포넌트의 store props로 설정해준 것이다.
Provider 컴포넌트를 이용하여 React 컴포넌트가 store에 access(접근)할 권한을 얻게 되었다면,
이제는 store에 저장되어 있는 state를 이용할 단계다.
각 컴포넌트들이 state를 이용하기 앞서서 store를 이용하기 위해서는
컴포넌트들을 store에 연결(connect)시켜야 한다. (state는 store에 저장되어 있기 때문)
그럼 컴포넌트를 store에 어떻게 연결(connect)시킬 수 있을까?
여기서 connect 함수를 사용한다.
connect 함수의 기본 구문은 다음과 같다.
connect(mapStateToProps, mapDispatchToProps)(Home);
connect 함수의 인자
- 첫번째 인자 mapStateToProps: 함수이다.
store로부터state를 가져와서 컴포넌트의props로 보내게 해준다. 자세한 용법은 3️⃣ mapStateToProps 참고- 두번째 인자 mapDispatchToProps:
dispatch를props로 보낼 수 있다. 자세한 용법은 4️⃣ mapDispatchToProp 참고- Home: 취득한 데이터를
props로 사용하고 싶은 컴포넌트를 지정한다.
그렇다면 실제로 connect 함수를 써보자.
템플릿 코드에서, Home 컴포넌트(Home.js)가 실질적으로 To-do(state)들을 렌더링하기 때문에
Home 컴포넌트를 store와 연결시킨다.
// routes/Home.js
import React, { useState } from "react";
import { connect } from "react-redux";
function Home() {
const [text, setText] = useState("");
function onSubmit(event) {
event.preventDefault();
console.log(text);
setText("");
}
function onChange(event) {
setText(event.target.value);
}
return (
<>
<h1>To-do List</h1>
<form onSubmit={onSubmit}>
<input
onChange={onChange}
type="text"
value={text}
placeholder="✍️Write To-do..."
/>
<button>✚</button>
</form>
<ul></ul>
</>
);
}
export default connect(mapStateToProps, mapDispatchToProps)(Home);
이제 connect 함수의 인수로 들어가는 mapStateToProps, mapDispatchToProps에 대해 각각 알아보자.
mapStateToPropsshould be defined as a function:function mapStateToProps(state, ownProps?)출처: React Redux 공식문서
mapStateToProps는 함수이며, connect 함수의 첫번째 인수이다.
mapStateToProps는 store로부터 state를 가져와서, 컴포넌트의 props로 state를 보내주는 역할을 한다.
즉, mapStateToProps를 사용한다는 것은, store로부터 데이터를 가져와서
그 데이터를 컴포넌트의 props에 넣는다는 뜻이다.
mapStateToProps의 인자
- 첫번째 인자 state:
store로부터 온state- 두번째 인자 ownProps: 생략가능. 컴포넌트가 현재 가지고 있는 모든
props를 보여준다
중요한 것은, mapStateToProps에서 return된 값이 컴포넌트의 props에 추가된다는 점이다.

위의 그림을 보면, mapStateToProps 함수에서 text: "hey"라는 값을 가진 object(객체)를 return시켰다.
그리고 connect 함수에서 연결했었던 Home 컴포넌트의 props를 확인한 결과
text: "hey"라는 값을 받고 있는 것이 보인다.
(그 외의 값은 react-router로부터 받은 props이다)
여기서 알 수 있는 것은, mapStateToProps 함수에서 store로부터 가져온 state를 return시킨다면
컴포넌트에서 state를 props로 받아서 사용할 수 있다는 것이다.
자, 그러면 우리의 To-do List 앱으로 돌아와서
store에서 받은 state를 컴포넌트에서 props로 받을 수 있도록
다음과 같이 코드를 작성한다.
// routes/Home.js
import React, { useState } from "react";
import { connect } from "react-redux";
function Home({ toDos }) {
console.log(toDos);
const [text, setText] = useState("");
function onSubmit(event) {
event.preventDefault();
console.log(text);
setText("");
}
function onChange(event) {
setText(event.target.value);
}
return (
<>
<h1>To-do List</h1>
<form onSubmit={onSubmit}>
<input
onChange={onChange}
type="text"
value={text}
placeholder="✍️Write To-do..."
/>
<button>✚</button>
</form>
<ul></ul>
</>
);
}
function mapStateToProps(state) {
return {
toDos: state
};
}
function mapDispatchToProps() {}
export default connect(mapStateToProps, mapDispatchToProps)(Home);
mapDispatchToProps는 connect 함수의 두번째 인자이며,
action을 reducer 함수에게 보내는 역할을 가진 dispatch를 props로 보낼 수 있다.
function mapDispatchToProp(dispatch, ownProps?)
mapDispatchToProp의 인자
- 첫번째 인자 dispatch:
Redux의store.dispatch()와 같음- 두번째 인자 ownProps: 생략가능. 컴포넌트가 현재 가지고 있는 모든
props를 보여준다
위에서 mapDispatchToProps의 첫번째 인자인 dispatch는
Redux의 store.dispatch()와 같다고 표현했는데,
그 이유는 다음의 그림과 같다.

mapDispatchToProps의 첫번째 인자의 dispatch를 그대로 return해주고,
Home 컴포넌트의 props으로 받은 dispatch를 보면
store.dispatch()와 같은 메소드가 들어있다는 것을 알 수 있다.
이렇게 mapDispatchToProp을 이용함으로써 컴포넌트 내에서 dispatch를 사용할 수 있게 되었다.
이제부터는 위에서 공부한 문법을 바탕으로 활용을 하는 것만 남았다.
To-do를 추가해서 페이지에 출력하기 위해서는 어떻게 해야할까?
input에 텍스트를 입력하여 submit(To-do를 추가하는 행위)하면
store.js의 addToDo라는 Action을 발동시켜야 한다.
addToDo를 발동시키기 전에, store.js를 다음과 같이 리팩토링하자.
// store.js
import { createStore } from "redux";
import { v4 as uuidv4 } from "uuid";
const ADD_TODO = "ADD_TODO";
const DELETE_TODO = "DELETE_TODO";
// Action Creator
const addToDo = (text) => {
return {
type: ADD_TODO,
text
};
};
// Action Creator
const deleteToDo = (id) => {
return {
type: DELETE_TODO,
id
};
};
const reducer = (toDos = [], action) => {
switch (action.type) {
case ADD_TODO:
return [{ text: action.text, id: uuidv4() }, ...toDos];
case DELETE_TODO:
return toDos.filter((toDo) => toDo.id !== action.id);
default:
return toDos;
}
};
const toDosStore = createStore(reducer);
// Action Creator의 묶음
export const actionCreators = {
addToDo,
deleteToDo
};
export default toDosStore;
Aciton creator인 addToDo와 deleteToDo를
actionCreators라는 object(객체)에 하나로 묶어주었다.
그 다음, 다시 Home.js로 돌아와서
본격적으로 To-do를 추가하기 위해 addToDo라는 Action을 발동시켜보자.
// routes/Home.js
import React, { useState } from "react";
import { connect } from "react-redux";
import { actionCreators } from "../store";
function Home({ toDos, addToDo }) {
const [text, setText] = useState("");
console.log(toDos);
function onSubmit(event) {
event.preventDefault();
addToDo(text);
setText("");
}
function onChange(event) {
setText(event.target.value);
}
return (
<>
<h1>To-do List</h1>
<form onSubmit={onSubmit}>
<input
onChange={onChange}
type="text"
value={text}
placeholder="✍️Write To-do..."
/>
<button>✚</button>
</form>
<ul>
{toDos.map((toDo) => (
<li key={toDo.id}>{toDo.text}</li>
))}
</ul>
</>
);
}
function mapStateToProps(state) {
return {
toDos: state
};
}
function mapDispatchToProps(dispatch) {
return {
addToDo: (text) => dispatch(actionCreators.addToDo(text))
};
}
export default connect(mapStateToProps, mapDispatchToProps)(Home);
mapDispatchToProps 함수는 object를 return하고 있다.
mapDispatchToProps 함수에서 return되는 object내의 addToDo라는 키를 가진 함수는 text를 받아야 하며,
dispatch는 actionCreators.addToDo를 받는다. actionCreators.addToDo 역시 text를 받는다.
mapDispatchToProps 함수에서 return되는 object는
Home 컴포넌트에서 props로 보내진다.
그렇게 props로 받아진 addToDo을 onSubmit에 추가해줌으로써
input을 submit할 때마다 addToDo라는 Action이 발동되게 했다.
이때 addToDo는 useState의 text를 받는다.
addToDo라는 Action이 발동되면 input의 입력값은 state(toDos)에 추가되고,
state(toDos)에 map을 사용하여 state(toDos)의 항목을 각각 출력해주었다.

다음은 각 To-do에 삭제 버튼을 추가하고,
To-do 버튼을 클릭했을 때 deleteToDo라는 Action을 발동시켜서
To-do를 state(toDos)로부터 삭제시켜야한다.
그 전에 먼저, 윗 단계에서 완성된 Home.js에서 각 To-do를 의미하는 <li>를
별도의 컴포넌트로 분리한 후 각 To-do에 삭제 버튼을 추가해보자. (코드가 커지기 때문)
// routes/Home.js
import React, { useState } from "react";
import { connect } from "react-redux";
import ToDo from "../components/ToDo";
import { actionCreators } from "../store";
function Home({ toDos, addToDo }) {
const [text, setText] = useState("");
function onSubmit(event) {
event.preventDefault();
addToDo(text);
setText("");
}
function onChange(event) {
setText(event.target.value);
}
return (
<>
<h1>To-do List</h1>
<form onSubmit={onSubmit}>
<input
onChange={onChange}
type="text"
value={text}
placeholder="✍️Write To-do..."
/>
<button>✚</button>
</form>
<ul>
{toDos.map((toDo) => (
<ToDo key={toDo.id} {...toDo} />
))}
</ul>
</>
);
}
function mapStateToProps(state) {
return {
toDos: state
};
}
function mapDispatchToProps(dispatch) {
return {
addToDo: (text) => dispatch(actionCreators.addToDo(text))
};
}
export default connect(mapStateToProps, mapDispatchToProps)(Home);
// components/ToDo.js
import React from "react";
function ToDo({ text, id }) {
return (
<>
<li id={id}>
{text}
<button>
<span role="img" aria-label="delete">
❌
</span>
</button>
</li>
</>
);
}
export default ToDo;
이제 ToDo의 삭제 버튼에 onClick 이벤트를 걸고
onClick 이벤트가 발동했을 때 (ToDo의 삭제 버튼을 클릭했을 때)
dispatch에서 deleteToDo라는 Action을 발동되게 해야 한다.
컴포넌트 내에서 dispatch를 이용하기 위해서 필요한 것은?
그렇다. connect 함수와 mapDispatchToProps이다.
이것을 당장 ToDo 컴포넌트에 사용해주자.
그리고 dispatch에서 deleteToDo라는 Action을 설정하는 것은
위의 ToDo를 Add할 때 기재했던 방식과 동일하다.
// components/ToDo.js
import React from "react";
import { connect } from "react-redux";
import { actionCreators } from "../store";
function ToDo({ text, deleteToDo }) {
return (
<>
<li>
{text}
<button onClick={deleteToDo}>
<span role="img" aria-label="delete">
❌
</span>
</button>
</li>
</>
);
}
function mapDispatchToProps(dispatch, ownProps) {
return {
deleteToDo: () => dispatch(actionCreators.deleteToDo(ownProps.id))
};
}
export default connect(null, mapDispatchToProps)(ToDo);
여기서 처음 눈에 띄는 것은 connect 함수의 첫번째 인자인 mapStateToProps 자리에 null이 작성 되어 있다.
이 ToDo 컴포넌트에서는 mapStateToProps를 사용할 필요가 없기 때문에,
(Home 컴포넌트로부터 props을 전달해받고있기 때문에)
mapStateToProps 자리에 null을 작성한 것이다.
그리고 눈에 띄는 것은 mapDispatchToProps의 두번째 인자인 ownProps이다.
4️⃣ mapDispatchToProp의 인자를 설명해주는 부분에서
ownProps는 컴포넌트가 현재 가지고 있는 모든 props를 보여준다고 했다.
ownProps를 확인해보자.

위의 Home 컴포넌트의 코드에서 ToDo 컴포넌트에게 state(toDos)의 각 내용을 {...ToDo}로 통째로 보내줬었다.
그렇게해서 현재 받은 props를 보여주는 것이 바로 ownProps이다.
ToDo를 삭제하는데에 필요한 deleteToDo Action은
id를 parameter(매개변수)로 필요로 하는데 (store.js의 deleteToDo 참고)
이 ownProps를 활용하면 이미 id 값을 가져올 수 있다.
그래서 deleteToDo 키값의 parameter(매개변수)에 id를 주지 않았고
dispatch의 actionCreators.deleteToDo에 바로 ownProps의 id를 주었다.
그 코드가 아래의 코드이다.
function mapDispatchToProps(dispatch, ownProps) {
return {
deleteToDo: () => dispatch(actionCreators.deleteToDo(ownProps.id))
};
}
이렇게 만들어진 deleteToDo는 ToDo 컴포넌트로 return되어 props로 보내지고,
props로 받은 deleteToDo를 삭제 버튼의 onClick 이벤트에 등록시키면...
function ToDo({ text, deleteToDo }) {
return (
<>
<li>
{text}
<button onClick={deleteToDo}>
<span role="img" aria-label="delete">
❌
</span>
</button>
</li>
</>
);
}

이렇게 ToDo를 삭제시키는 것에 성공했다!🙌
지금까지 To-do를 추가하고 삭제하는 기본적인 동작까지 구현하는 것에 성공했다.
마지막으로, React Router를 이용해 ToDo 컴포넌트의 각 ToDo에
Detail 페이지의 링크를 걸어서 To-do의 페이지에 접속해보며
Detail 페이지에 To-do와 To-do의 id를 출력해보자.
먼저 각 ToDo를 클릭하면 Detail 페이지로 이동할 수 있도록
React Router를 이용해 ToDo 컴포넌트의 각 ToDo에
Detail 페이지의 링크를 걸자.
(Router 설정은 템플릿 코드의 component/App.js에서 미리 진행해두었다.)
// components/ToDo.js
import React from "react";
import { connect } from "react-redux";
import { Link } from "react-router-dom";
import { actionCreators } from "../store";
function ToDo({ text, deleteToDo, id }) {
return (
<>
<li>
<Link to={`/${id}`}>{text}</Link>
<button onClick={deleteToDo}>
<span role="img" aria-label="delete">
❌
</span>
</button>
</li>
</>
);
}
function mapDispatchToProps(dispatch, ownProps) {
return {
deleteToDo: () => dispatch(actionCreators.deleteToDo(ownProps.id))
};
}
export default connect(null, mapDispatchToProps)(ToDo);
Detail 페이지의 parameter는 각 To-do의 id로 설정하기 위해
ToDo 컴포넌트의 props로 가져온 id를 Link에 설정했다.
이제 Detail 페이지에 To-do와 To-do의 id를 출력할 차례이다.
To-do를 출력하기 위해서는 state에 저장된 ToDo를 불러와야 한다.
컴포넌트 내에서 state를 가져와서 state를 props로 보낼 때 사용하는 것은?
3️⃣ mapStateToProps이다.
Detail 컴포넌트에서 mapStateToProps를 불러오자.
// routes/Detail.js
import React from "react";
import { connect } from "react-redux";
function Detail(state) {
return (
<>
<h1>Detail</h1>
<p>id</p>
</>
);
}
function mapStateToProps(state, ownProps) {
console.log(ownProps)
return {
state
};
}
export default connect(mapStateToProps)(Detail);
mapStateToProps의 두번째 인자인 ownProps를 이용해서 Detail 컴포넌트의 현재 props를 확인해보자.

Detail 컴포넌트는 components/App.js에서 React Router의 Route로써 사용되고 있다.
Detail 컴포넌트처럼 Route로 사용된 컴포넌트는 props로 history, match, location라는 3개의 object(객체)를 받을 수 있게 된다.
history, match, location라는 각각의 object(객체)는
URL path, 페이지 경로명, 페이지 탐색 history 등과 같은 정보를 담고 있다.
history, match, location가 담고 있는 정보들에 대해서는
이 블로그에서 자세한 정보를 얻을 수 있다.
우리의 To-do List 앱에서는
다양한 ToDo가 저장되어있는 state 중에서
현재 접속한 Detail 페이지의 URL parameter(id)와 같은 id를 가진 그 ToDo가 가진 정보를 출력할 것이다.
이때 현재 접속한 Detail 페이지의 URL parameter(id)를 찾기 위해
match object(객체)의 정보를 이용하려 한다.
// routes/Detail.js
import React from "react";
import { connect } from "react-redux";
function Detail({ toDo }) {
return (
<>
{/* toDo뒤의 ?는 Optional Chaining이라는 문법임.
null이나 undefined인 값이 반환되면, 코드를 즉시 중단하고 undefined를 반환함 */}
<h1>{toDo?.text}</h1>
<p>{toDo?.id}</p>
</>
);
}
function mapStateToProps(state, ownProps) {
// App.js에서 Detail Route에 설정한 동적 라우팅(path="/:id")으로 전달된
// path 파라미터 정보를 불러옴
const {
match: {
params: { id }
}
} = ownProps;
return {
// path 파라미터의 id와 같은 toDo.id를 찾음
toDo: state.find((toDo) => toDo.id === id)
};
}
export default connect(mapStateToProps)(Detail);
여기서 중요한 포인트는 toDo를 출력하는데에 Optional Chaining이라는 문법을 사용했다.
Optional Chaining에 대하여 (MDN 문서)
이번에는 간단한 React 프로젝트에서 React Redux를 배워보는 시간을 가졌다.
Redux는 이해하는데에 약간 비용이 들긴 하지만
(무엇이든 배우는 데에는 시간과 노력이 필요하긴 하다!)
Redux를 사용함으로써 코드가 정리된 듯한 느낌이 든다.
좀 더 연습해서 숙련될 수 있도록 노력해야겠다💪
좋은글 감사합니다 이해하는데 많은 도움되었어요!