[React] Redux toolkit
๐ Redux toolkit
๋ฆฌ์กํธ ํ๊ฒฝ์๋ ์๋ง์ ์ํ๊ด๋ฆฌ ๋ฐฉ๋ฒ์ด ์์ต๋๋ค.
๋ฆฌ์กํธ์ ์ด๋ฏธ ๋ด์ฅ๋์ด ์๋ Context API๊ฐ ์์ผ๋ฉฐ, ์ด์ธ์๋ ๋ค์ํ ์ํ ๊ด๋ฆฌ ๋ผ์ด๋ธ๋ฌ๋ฆฌ๋ค์ด ์กด์ฌํฉ๋๋ค.
๊ทธ๋ผ์๋ ๋ถ๊ตฌํ๊ณ ๋ฆฌ๋์ค๊ฐ ๋ง์ ๊ฐ๋ฐ์๋ค์๊ฒ ์ฌ์ฉ๋๊ณ ์๋ ์ด์ ๋ ๋ฌด์์ผ๊น์.
๐ Context API vs Redux
์ฌ์ค Context API์ Redux๋ ๋น๊ต๋์์ด ์๋๋๋ค.
Context๋ ์๋จ์ผ ๋ฟ, ์ํ๊ด๋ฆฌ ์์ฒด๋ useState
์ useReducer
๊ฐ ๋ด๋นํฉ๋๋ค.
Context API์ Redux์ ๊ฐ์ฅ ํฐ ์ฐจ์ด์ ์ ์ฑ๋ฅ ์ต์ ํ์์ ๋ณด์ฌ์ง๋๋ค.
Context API๋ ์ฑ๋ฅ ์ต์ ํ๊ฐ ์ด๋ฃจ์ด์ง์ง ์์ ํน์ ๊ฐ์ ์์กดํ๋ ๊ฒฝ์ฐ, ํด๋นํ๋ ํน์ ๊ฐ์ด ์๋ ๋ค๋ฅธ ๊ฐ์ด ๋ณ๊ฒฝ๋ ๋๋ ์ปดํฌ๋ํธ๊ฐ ๋ฆฌ๋ ๋๋ง์ด ๋ฉ๋๋ค.
๋ฐ๋ฉด Redux๋ ์์กดํ์ง ์๋ ๊ฐ์ด ๋ฐ๋๊ฒ๋๋ฉด ๊ทธ ๊ฐ์ ์ํฅ์ ๋ฐ์ง ์๊ณ ๋ฆฌ๋ ๋๋ง์ด ๋ฐ์ํ์ง ์์ต๋๋ค.
๋ฐ๋ผ์ Context API๋ฅผ ์ฌ์ฉํ ๊ฒฝ์ฐ์๋ ์ปดํฌ๋ํธ์ ๋ถ๋ฆฌ๊ฐ ์ค์ํฉ๋๋ค.
๐ Redux tookit ์์ํ๊ธฐ
โช Installation
โช CRA๋ฅผ ์ฌ์ฉํ๋ ๊ฒฝ์ฐ
# Redux + Plain JS template
npx create-react-app my-app --template redux
# Redux + TypeScript template
npx create-react-app my-app --template redux-typescript
โช ์กด์ฌํ๋ ์ฑ์ ์ถ๊ฐํ๋ ๊ฒฝ์ฐ
# NPM
npm install @reduxjs/toolkit
# Yarn
yarn add @reduxjs/toolkit
๐ Redux toolkit์ ์กด์ฌํ๋ API
โช configureStore()
createStores
๋ ๊ฐ๋จํ configuration ์ต์ ๋ค๊ณผ ๊ธฐ๋ณธ๊ฐ์ ์ ๊ณตํ๋ค.- ์๋์ผ๋ก ์ฌ์ฉ์์ slice reducers๋ฅผ ํฉ์ณ์ ์ฌ์ฉํ๋ middleware๊ฐ ๋ฌด์์ด๋ ์ถ๊ฐํ๋ค.
โช createReducer()
- ์ํ๊ฐ ๋ฐ๋๊ธฐ ์ ์ ์ฌ์ฉ์๊ฐ ๋ฃฉ์
ํ
์ด๋ธ์ ์ ๊ณตํ๊ฒ ํ๋ค.
lookup table? ์ฃผ์ด์ง ์ฐ์ฐ์ ๋ํด ๋ฏธ๋ฆฌ ๊ณ์ฐ๋ ๊ฒฐ๊ณผ๋ค์ ์งํฉ(๋ฐฐ์ด)
- ์๋์ผ๋ก
immer
๋ผ์ด๋ธ๋ฌ๋ฆฌ๋ฅผ ์ฌ์ฉํด์ ๊ฐ๋จํ immutable ์ ๋ฐ์ดํธ๋ฅผ ์์ฑํ๊ฒ ํ๋ค.
โช createAction()
- ์ฃผ์ด์ง ์ก์ ํ์ ์ ๋ํด stringํ์ ์ผ๋ก ์ก์ ํจ์๋ฅผ ๋ฐ์์ํจ๋ค.
- ์ด ํจ์๋
toString()
์ด ๋ด์ฅ๋์ด ์์ด, ์์ ํ์ ์ ๋์ ํด์ ์ฐ์ธ๋ค.
โช createSlice()
- reducer ํจ์๋ค, slice ์ด๋ฆ, ์ด๊ธฐ state ๊ฐ์ ํ์ฉํ์ฌ ์๋์ผ๋ก slice reducer๋ฅผ ์์ฑํ๋ค.
โช createAsyncThunk
- string ํ์ ์ ์ก์ ๊ณผ ํ๋ก๋ฏธ์ค๋ฅผ ๋ฐํํ๋ ํจ์๋ฅผ ๋ฐ์๋ค์ธ๋ค.
- ์๋์ผ๋ก slice reducer๋ฅผ ์์ฑํ๋ค.
โช createEntitiyAdapter
- ์ผ๋ จ์ ์ฌ์ฌ์ฉ ๊ฐ๋ฅํ ๋ฆฌ๋์์ ์คํ ์ด ์์ ๋ฐ์ดํฐ๋ฅผ ๊ด๋ฆฌํ๊ธฐ ์ํ selectors๋ฅผ ์์ฑํ๋ค.
๐ RTK Query
RTK Query๋ @reduxjs/toolkit
ํจํค์ง์ ์ถ๊ฐ๋์ด ์๋ ๋ถ๊ฐ์ ์ธ ์ต์
์
๋๋ค.
API ์ธํฐํ์ด์ค๋ฅผ ์ ์ํ๊ธฐ ์ํด ๋ฐ์ดํฐ๋ฅผ fetchingํ๊ณ cachingํ๋ ๊ฐ๋จํ์ง๋ง ๊ฐ๋ ฅํ ํด์ ์ ์ ๊ณตํฉ๋๋ค.
๐ RTK Query์ ํฌํจ๋ API๋ค
โช createApi()
- RTK Query์ ํต์ฌ์ ์ธ ๊ธฐ๋ฅ
- ๋ฐ์ดํฐ๋ฅผ fetchingํ๊ณ ๋ณํํ๋ ๋ฐฉ๋ฒ์ ํฌํจํด์ ์๋ํฌ์ธํธ๋ฅผ ์ ์ํ๊ณ ์๋ํฌ์ธํธ๋ก๋ถํฐ ๋ฐ์ดํฐ๋ฅผ ๊ฒ์ํ๋ ๋ฐฉ๋ฒ์ ํ์ฉํ๋ค.
endpoint? ์ปดํจํฐ ๋คํธ์ํฌ์ ์ฐ๊ฒฐํ๊ณ ์ปดํจํฐ ๋คํธ์ํฌ์์ ๋ณด๋ฅผ ๊ตํํ๋ ๋ฌผ๋ฆฌ์ ๋๋ฐ์ด์ค
โช fetchBaseQuery()
- ์์ฒญ์ ๋จ์ํํ๋ ๊ฒ์ ๋ชฉํ๋ก ํ๋
fetch
๋ฅผ ๊ฐ์ผ๋ค. = ๋ฐ์ดํฐ fetching์ ๋จ์ํํ๊ธฐ ์ํด ์ฌ์ฉ๋๋ค. - Redux toolkit ๊ณตํ์
createApi
์์baseQuery
๋ฅผ ์ฌ์ฉํ๋ ๊ฒ์ ์ถ์ฒํ๋ค.
โช ApiProvider
- ์์ง Redux Store๊ฐ ์๋ ๊ฒฝ์ฐ ์ฌ์ฉํ ์ ์๋ค.
โช setupListeners()
refetOnMount
์refetchOnReconnect
๋ฅผ ์ํด ์ฌ์ฉ๋๋ ์ ํธ๋ฆฌํฐ
๐ ํ์ ์คํฌ๋ฆฝํธ์ RDK ์์ํ๊ธฐ
โช Redux Toolkit ์ด๊ธฐ ์ค์
๐พ app/store.ts
import { configureStore, ThunkAction, Action } from "@reduxjs/toolkit";
import counterReducer from "../features/counter/counterSlice";
export const store = configureStore({
reducer: {
counter: counterReducer,
},
});
export type AppDispatch = typeof store.dispatch;
// ํ์
์๋ฌ๋ฅผ ๋ง๊ธฐ ์ํด ์คํ ์ด ์ค์ ํ์ผ์์ ์ง์ ๋ด๋ณด๋ด๊ณ ๋ค๋ฅธ ํ์ผ๋ก ์ง์ ๊ฐ์ ธ์ค๋ ๊ฒ์ด ์์ ํ๋ค.
export type RootState = ReturnType<typeof store.getState>;
export type AppThunk<ReturnType = void> = ThunkAction<
ReturnType,
RootState,
unknown,
Action<string>
>;
๐พ app/hooks.ts
์ฌ์ฉํ๋ ํ
๋ค์ ํ์
์ ์ง์ ํด์ผ ํฉ๋๋ค.
RootState
์ AppDispatch
์ ํ์
์ ๊ฐ๊ฐ ์ปดํฌ๋ํธ์์ importํด์ ์ฌ์ฉํ๋ ๊ฒ์ด ๊ฐ๋ฅํ์ง๋ง,
useDispatch
์ useSelector
๋ ํ์
์ด ์ง์ ๋ ํ
์ผ๋ก ์ฌ์ฉํ๋ ๊ฒ์ด ๋ ์ข์ต๋๋ค.
useSelector
์ ๊ฒฝ์ฐ, ๋งค๋ฒ(state: RootState)
์ ํ์ ์ ์ง์ ํด์ค ํ์๊ฐ ์์ต๋๋ค.useDispatch
์ ๊ฒฝ์ฐ, ๊ธฐ๋ณธDispatch
๋ thunk๋ฅผ ์์ง๋ชปํด์ thunk middleware ํ์ ์ด ํฌํจ๋ ์คํ ์ด์์ ์ปค์คํฐ๋ง์ด์ง๋AppDispatch
ํ์ ์ ์ฌ์ฉํด์ผํฉ๋๋ค.useDispatch
๋ฅผ ์ถ๊ฐํ๋ฉด ํ์ํ ๋AppDispatch
๋ฅผ ๊ฐ์ ธ์ฌ ์ ์์ต๋๋ค.
์ด ํ ๋ค์ ํ์ ์ด ์๋๋ผ ๋ณ์์ด๊ธฐ ๋๋ฌธ์ store ํ์ผ์ด ์๋๋ผ hooksํ์ผ์ ์ง์ ํด์ผํ๋ค. ์ด๋ ๊ฒ hooksํ์ผ์ ์ง์ ํด์ ํ์ํ ๋ ๋ง๋ค componentํ์ผ์์ importํด์ฌ ์ ์๋ค.
import { TypedUseSelectorHook, useDispatch, useSelector } from "react-redux";
import type { RootState, AppDispatch } from "./store";
// Use throughout your app instead of plain `useDispatch` and `useSelector`
export const useAppDispatch = () => useDispatch<AppDispatch>();
export const useAppSelector: TypedUseSelectorHook<RootState> = useSelector;
๐ Application ์ฌ์ฉ
โช Slice State์ Action Types
๊ฐ๊ฐ slice ํ์ผ์ ์ด๊ธฐ state value์ ๋ํ ํ์
์ ์ง์ ํด์ createSlice
๊ฐ ๊ฐ๊ฐ ๋ฆฌ๋์์ state
์ ๋ํ ํ์
์ ์ถ๋ก ํ ์ ์์ต๋๋ค.
initial state type์ ๋ฏธ๋ฆฌ ์ง์ ํ๊ณ initial state ๊ฐ์ฒด๋ฅผ ์์ฑํฉ๋๋ค.
// slice state ํ์
์ ์
export interface CounterState {
value: number;
status: "idle" | "loading" | "failed";
}
// initial state ๊ฐ์ฒด ์์ฑ
const initialState: CounterState = {
value: 0,
status: "idle",
};
โช createSlice๋ก slice ์์ฑ
createSlice๋ name, initialState, reducers๊ฐ ์์ต๋๋ค.
name
: action์์ ๋ถ์ด ๋ค๋ฅธ slice์ action๋ค๊ณผ ์ค๋ณต์ ํผํ๋ค.initialState
: ๋ฏธ๋ฆฌ ์์ฑํ initialState๊ฐ ๋ค์ด์๋ค.reducer
: reducer๋ action ์ญํ ์ ํ๊ณ state์ ๋ณํ๋ฅผ ๋ด๋นํ๋ค.immer.js
๋ฅผ ๋ด์ฅํ๊ณ ์์ด state๊ฐ์ ์๋์ผ๋ก returnํ๋ค.
const counterSlice = createSlice({
name: "counter",
initialState,
reducers: {
increment: (state) => {
state.value += 1;
},
decrement: (state) => {
state.value -= 1;
},
incrementByValue: (state, action: PayloadAction<number>) => {
state.value += action.payload;
},
},
});
โช export
slice ๋ด์ actions๊ณผ reducer๋ฅผ exportํฉ๋๋ค.
// export actions
export const { increment, decrement, incrementByValue } = counterSlice.actions;
// export default slice.reducer
export default counterSlice.reducer;
โช Store ์์ฑ
store์๋ state์ dispatchํ ํจ์๋ค์ด ๋ค์ด์์ต๋๋ค.
ํ์ ์คํฌ๋ฆฝํธ๋ฅผ ์ฌ์ฉํ๊ณ ์๋ค๋ฉด ๊ฐ๊ฐ state์ dispatch์ ํ์ ์ ์ง์ ํด ์ฃผ์ด์ผ ํ๋ค.
- store ์์ฑ ๋ช
๋ น์ด:
configureStore
import { configureStore } from "@reduxjs/toolkit";
import CounterReducer from "../features/counter/counter";
export const store = configureStore({
reducer: {
counter: CounterReducer,
},
});
// store์ dispatch์ ํ์
export type RootState = ReturnType<typeof store.getState>;
export type AppDispatch = typeof store.dispatch;
โช Provider ์์ฑ
Provider์ store์ app์ ์ฐ๊ฒฐํด์ ์ปดํฌ๋ํธ๋ค์ด store์ ์๋ state๋ dispatch๋ฅผ ์ฌ์ฉํ ์ ์๊ฒ ํฉ๋๋ค. ๊ฐ์ฅ ์์ ์ปดํฌ๋ํธ์ store๋ฅผ ์ฐ๊ฒฐํ๋ฉด ํ์ ์ปดํฌ๋ํธ์์๋ store๋ฅผ ์ฌ์ฉํ ์ ์์ต๋๋ค.
// index.tsx
import { Provider } from "react-redux";
import { store } from "./app/store";
ReactDOM.render(
<React.StrictMode>
<Provider store={store}>
<App />
</Provider>
</React.StrictMode>,
document.getElementById("root")
);
โช ์ปดํฌ๋ํธ ์์์ ์ฌ์ฉํ๊ธฐ
import React, { useState } from "react";
import { useAppDispatch, useAppSelector } from "../../app/hooks";
import { increment, decrement, incrementByValue } from "./counter";
function CounterView() {
// ์ค์ ํ hook๋ค์ ์ ์ฉ
const count = useAppSelector((state) => state.counter.value);
const dispatch = useAppDispatch();
// useState๋ก num๊ฐ ๊ด๋ฆฌ
const [num, setNum] = useState<number>(0);
const handleChange = (e: React.ChangeEvent<HTMLInputElement>) => {
setNum(parseInt(e.currentTarget.value));
};
return (
<>
<div
style=
>
// ๋ฒํผ์ด ๋๋ฆฌ๋ฉด importํด์จ ํจ์๋ค์ด dispatch๋๋ค.
<button onClick={() => dispatch(decrement())}>-1</button>
<h1 style=>{count}</h1>
<button onClick={() => dispatch(increment())}>+1</button>
</div>
<div>
<input type="number" onChange={handleChange} />
<button onClick={() => dispatch(incrementByValue(num))}>+{num}</button>
</div>
</>
);
}
export default CounterView;
๐ฌ ์ต์ ๋๊ธ