Redux
Redux란?
- JavaScript 앱의 상태 관리를 위한 라이브러리 중 하나
- 앱의 상태를 중앙 집중식으로 관리하고 상태 변경을 예측 가능하게 만들어 앱의 복잡성을 관리
- 해당 React 프로젝트 경로 터미널에
npm install redux
,npm install react-redux
입력하여 설치
Redux Flux 패턴
- Redux는 단방향으로 데이터를 전달하는 패턴 방식
Store
- 앱의 상태를 저장하고 관리하는 저장소
- 저장소는 단 한개만 존재
Reducer
- 상태 변경을 처리하는 순수한 함수
- 현재 state와 action을 받아서 새로운 상태를 반환
Action
- 상태 변경을 나타내는 객체
- action은
type
속성과 선택적으로 데이터(payload)를 가진다 - dispatch 함수를 통해 Reducer로 전달
Dispatch
- action의 type을 dispatch하면 Reducer를 호출하여 상태를 업데이트 후, store에 저장
Redux 구현
- 구현
- "1씩 증가": 값을 하나씩 증가하여 화면 출력
- "5씩 증가": 값을 5씩 증가하여 화면 출력
- "-1씩 감소": 값을 -1씩 감소하여 화면 출력
- "숫자 가리기/켜기": 버튼을 클릭시, 표현하고 있는 숫자 가리기/켜기
store 생성 및 component에 전달
createStore()
: redux에서 제공하는 함수, Redux store를 생성하기 위해 사용하는 함수
import React from "react";
import { createRoot } from "react-dom/client";
import { Provider } from "react-redux";
import rootReducer from "./store/store";
import { createStore } from "redux";
import Button from "./components/Button";
const store = createStore(rootReducer);
const container = document.getElementById("root");
const root = createRoot(container);
root.render(
<Provider store={store}>
<Button />
</Provider>
);
store.js에서 생성한 Reducer를 createStore()
를 통해 저장소를 생성한다.
<Provider>
를 이용하여 하위 Component들에게 Redux store를 전달한다.
즉, <Provider>
안에 감싸진 하위 Component간의 상태 공유 및 업데이트가 가능하다.
Reducer 생성 및 결합
combineReducers()
: redux에서 제공하는 함수, Reducer들을 조합하여 각 Reducer가 관리하는 일부 상태를 모아 하나의 큰 상태 객체로 만드는 것
import { combineReducers } from "redux";
const countInitialState = { count: 0 };
const counterReducer = (state = countInitialState, action) => {
if (action.type === "increase") return { ...state, count: state.count + 1 };
else if (action.type === "increase5")
return { ...state, count: state.count + action.payload };
else if (action.type === "decrease")
return { ...state, count: state.count - 1 };
else return state;
};
const visibleInitialState = { visible: true };
const visibleReducer = (state = visibleInitialState, action) => {
switch (action.type) {
case "on/off":
return { visible: !state.visible };
default:
return state;
}
};
const rootReducer = combineReducers({ counterReducer, visibleReducer });
export default rootReducer;
store.js는 저장소를 의미하며, 내부적으로 어떻게 동작할지에 대해 정의를 한 코드이다.
countInitialState
와 visibleInitialState
을 통해 각 변수의 속성 값을 초기화한다.
counterReducer
와 visibleReducer
는 각 Redux Reducer함수라고 정의를 한다.
각 함수에는 state와 action이라는 Parameter가 존재한다.
state
는 각 Reducer에서 관리하고 있는 객체의 속성 상태를 표현action
은 Redux에서 상태 변경이 요청됨에 따라 state의 상태를 변경
Parameter 값 중 state = 초기값
을 하는 이유는 처음 Mount가 될 경우 undefined 값을 방지하기 위해 사용된다.
const counterReducer = (state = countInitialState, action) => {
if (action.type === "increase") return { ...state, count: state.count + 1 };
else if (action.type === "increase5")
return { ...state, count: state.count + action.payload };
else if (action.type === "decrease")
return { ...state, count: state.count - 1 };
else return state;
};
const dispatch = useDispatch();
const increase1 = () => {
dispatch({ type: "increase" });
};
Button.jsx에서 increase1에 대한 이벤트가 발생 시, useDispatch
를 이용하여 action에 대한 객체 정보를 counterReducer로 보낸다.
그럼 increase1의 함수를 이벤트 호출하였다고 가정을 하였을 때,
{ type: "increase" }
의 값은 counterReducer 함수의 action에 들어간다.
action안의 type이라는 객체 속성을 이용하여 state의 정보를 상태 변경 후, store에 저장을 한다.
const rootReducer = combineReducers({ counterReducer, visibleReducer });
export default rootReducer;
store.js에는 총 2개의 Reducer인 counterReducer와 visibleReducer가 존재한다.
Redux의 store는 한개의 저장소만 존재해야하기 때문에 Reducer들을 결합하여야한다.
combineReducers({counterReducer, visibleReducer})
를 통하여 하나의 Reducer로 정의를 하며,
index.js에서 store를 생성하기 때문에 rootReducer를 export 하였다.
state 데이터 가져오기 및 action 전달
useSelector()
: Redux Toolkit에서 제공하는 함수, Redux store의 상태를 읽어오며, 선택하는데 사용useDispatch()
: Redux Toolkit에서 제공하는 함수, action을 Conponent 내부에서 발생시키고, Redux store에 action의 type 및 추가 데이터를 전달
import React from "react";
import { useDispatch, useSelector } from "react-redux";
function Button() {
const { count } = useSelector((state) => state.counterReducer);
const { visible } = useSelector((state) => state.visibleReducer);
const dispatch = useDispatch();
const increase1 = () => {
dispatch({ type: "increase" });
};
const increase5 = () => {
dispatch({ type: "increase5", payload: 5 });
};
const decrease = () => {
dispatch({ type: "decrease" });
};
const isVisible = () => {
dispatch({ type: "on/off" });
};
return (
<div>
{visible && <p>{count}</p>}
<button onClick={increase1}>1씩 증가</button>
<button onClick={increase5}>5씩 증가</button>
<button onClick={decrease}>-1씩 감소</button>
<button onClick={isVisible}>숫자 {visible ? "가리기" : "켜기"}</button>
</div>
);
}
export default Button;
Button.jsx는 Redux store에 있는 data들을 사용하여 기능을 구현하는 코드이다.
const { count } = useSelector((state) => state.counterReducer);
const { visible } = useSelector((state) => state.visibleReducer);
useSelector의 state에 대 한 출력 결과물이다.
즉, useSelector는 Redux store에 있는 전체 상태 객체에서 원하는 데이터를 선택한다.
각 Redux store 안에 있는 Reducer들의 state값을 구조분해 할당으로 가져온 것이다.
상태가 변할때 마다 랜더링이 되어 각 state값을 새로 할당한다.
const dispatch = useDispatch();
const increase1 = () => {
dispatch({ type: "increase" });
};
const increase5 = () => {
dispatch({ type: "increase5", payload: 5 });
};
const decrease = () => {
dispatch({ type: "decrease" });
};
const isVisible = () => {
dispatch({ type: "on/off" });
};
useDispatch
를 이용하여 각 이벤트 핸들러가 호출되었을 때의 Redux의 action을 store에 전달하여 해당 Reducer를 실행한다.
만약 "1씩 증가"라는 버튼을 눌렀을 경우, dispatch({type: "increase"})
가 실행된다.
그럼, store.js의 counterReducer함수에게 action 객체 정보 를 store에 전달 후, 해당 Reducer에 전달된다.
이후 counterReducer함수에서 state 업데이트 및 store에 저장 한다.
store의 저장된 이후, Component가 렌더링이 되어 상태 변경이 일어난 뒤의 값을 호출한다.
const dispatch = useDispatch();
const increase5 = () => {
dispatch({ type: "increase5", payload: 5 });
};
dispatch를 통하여 { type: "increase5", payload: 5 }
의 정보를 Redux store에 전달 후, 해당 Reducer에게 action을 보낸다.
payload는 action객체의 type 이외에 추가할 데이터를 의미
const counterReducer = (state = countInitialState, action) => {
// if (action.type === "increase") return { ...state, count: state.count + 1 };
else if (action.type === "increase5"){
console.log(action)
return { ...state, count: state.count + action.payload };
}
// else if (action.type === "decrease")
// return { ...state, count: state.count - 1 };
// else return state;
};
위와 같이 "5씩 증가" 버튼을 눌렀을 때, 해당 Reducer의 action에 type과 payload의 출력물 확인할 수 있다.
counterReducer에 전달된 action 객체의 type속성으로 해당하는 state 정보를 업데이트 후, store에 저장한다.