상태 관리라는 주제는 RN에 국한된 내용은 아니다. typescript를 기반으로 여러 방법을 복습겸 테스트 해보자. 더 나아가 아직 사용해보지 않는 recoil 라이브러리도 경험해 보자.
아래에서 배운 내용은 대부분 생략한다.
Getting Start
$ npx react-native init StateManagementInReactNative --template react-native-template-typescript
1. Redux Toolkit (RTK)
typescript에서 redux를 사용하려고 하니, 너무 많은 타이핑과 구조체들이 필요해서 실제로 프로덕션 환경에서 도입하여 사용해본 반가운 라이브러리다. (참고: 공식문서)
오랫만에 다시 보자 리덕스의 개념 및 데이터 흐름!
설치
$ yarn add redux react-redux @reduxjs/toolkit
$ yarn add -D @types/react-redux
기본 구현체
사용자의 인증 상태를 관리할 authSlice 생성.
// slices/auth.ts
import {createSlice, PayloadAction} from '@reduxjs/toolkit';
export interface User {
id: number;
username: string;
displayName: string;
}
interface AuthState {
user: User | null;
}
const initialState: AuthState = {
user: null,
};
const authSlice = createSlice({
name: 'auth',
initialState,
reducers: {
authorize(state, action: PayloadAction<User>) {
state.user = action.payload;
},
logout(state) {
state.user = null;
},
},
});
export default authSlice.reducer;
export const {authorize, logout} = authSlice.actions;
rootReducer 생성
// slices/index.ts
import {combineReducers} from 'redux';
import auth from './auth';
const rootReducer = combineReducers({
auth,
});
export type RootState = ReturnType<typeof rootReducer>;
export default rootReducer;
스토어 추가
import React from 'react';
import {Provider} from 'react-redux';
import {createStore} from 'redux';
import rootReducer from './slices';
const store = createStore(rootReducer);
function App() {
return <Provider store={store}>{/* ... */}</Provider>;
}
export default App;
상태 가져오기
useSelector 사용
function AuthStatus() {
const user = useSelector((state: RootState) => state.auth.user);
return (
<View style={styles.status}>
<Text style={styles.text}>
{user != null ? user.displayName : '로그인하세요'}
</Text>
</View>
);
}
액션 함수 전달하기
useDispatch 훅으로 dispatch 함수를 얻고 필요할때 dispatch 함수로 액션 함수 전달. (다시 한번 더 읽기 리덕스의 핵심 개념 및 원리!)
function AuthButtons() {
const dispatch = useDispatch();
const onPressLogin = () => {
dispatch(
authorize({
id: 1,
username: 'JungKyuHyun',
displayName: 'Jacob',
}),
);
};
const onPressLogout = () => {
dispatch(logout());
};
return (
<View>
<Button title="로그인" onPress={onPressLogin} />
<Button title="로그아웃" onPress={onPressLogout} />
</View>
);
}
useSelector 사용시 RootState 없이 타입 추론되게 하기
다시 공부하면서 발견한 내용이다. 매번 RootState 타입을 가져왔었는데 이런 방법이 존재했네;;
// slices/index.ts
// ...
export type RootState = ReturnType<typeof rootReducer>;
// <-- here! --->
declare module 'react-redux' {
interface DefaultRootState extends RootState {}
}
export default rootReducer;
리덕스와 연동하는 로직을 Hook으로 분리
리덕스의 상태를 다루는 로직과 컴포넌트의 UI로직을 분리하자. (나중에 상태 관리 라이브러리 마이그레이션할 때도 편하며, 커스텀 훅으로 만들면 재사용도 가능해 진다.)
먼저 AuthStatus.tsx에서 사용했던 스토어로부터 상태를 가져오던 부분을 커스텀 훅으로 만들었다.
// hooks/useUser.ts
export default function useUser() {
return useSelector(state => state.auth.user);
}
AuthButtons.tsx에서 사용하던 리덕스 부분을 분리했다.
// hooks/useAuthActions.ts
// 1번 방법
export default function useAuthActions() {
const dispatch = useDispatch();
return {
authorize: (user: User) => dispatch(authorize(user)),
logout: () => dispatch(logout()),
};
}
1번 방법은 직접 타이핑 해보면 알겠지만, dispatch하기 위해 한번 더 랩핑하는 과정이 귀찮다.
// hooks/useAuthActions.ts
// 2번 방법
export default function useAuthActions() {
const dispatch = useDispatch();
return bindActionCreators({authorize, logout}, dispatch);
}
2번 방법의 경우 제공되는 유틸함수 이용(bindActionCreators)를 이용했다. 훨씬도 간결하고 깔끔해 졌다.
또한 각 액션 생성 함수들의 파라미터 타입을 따로 알지 않아도 되기 때문에 편하다.
// hooks/useAuthActions.ts
// 3번 방법
export default function useAuthActions() {
const dispatch = useDispatch();
return useMemo(
() => bindActionCreators({authorize, logout}, dispatch),
[dispatch],
);
}
3번 방법은 2번 방법을 최적화한 것이다.
이 훅에서는 컴포넌트가 새로 렌더링될 때마다 bindActionCreators가 호출되어 각 함수들이 새로 선언되게 되는데, 그 함수가 useEffect에서 사용하게 되면 의도치 않은 버그가 생길 수 있기 때문에 useMemo를 사용해 준다.
리덕스로 할일 목록 만들기
Redux Toolkit을 사용할 때는 불변성을 유지하지 않아도 자동으로 관리되기 때문에 push, slice 등의 함수를 사용해도 괜찮다.
다만 리듀서에서 반환하는 값이 없을 때는 라이브러리에서 불변성 유지를 자동으로 해주지만, 값을 반환한다면 불변성 자동 관리가 생략된다.
import {createSlice, nanoid, PayloadAction} from '@reduxjs/toolkit';
export interface Todo {
id: string;
text: string;
done: boolean;
}
const initialState: Todo[] = [
{id: '1', text: 'RN 배우기', done: true},
{id: '2', text: 'React 배우기', done: false},
{id: '3', text: 'React Redux 배우기', done: false},
];
const todoSlice = createSlice({
name: 'todos',
initialState,
reducers: {
add: {
// @see https://redux-toolkit.js.org/api/createSlice#customizing-generated-action-creators
prepare(text: string) {
return {payload: {id: nanoid(), text}};
},
reducer(state, action: PayloadAction<{id: string; text: string}>) {
state.push({
...action.payload,
done: false,
});
},
},
remove(state, action: PayloadAction<string>) {
return state.filter(todo => todo.id !== action.payload);
},
toggle(state, action: PayloadAction<string>) {
const selected = state.find(todo => todo.id === action.payload);
if (selected === undefined) {
return;
}
selected.done = !selected.done;
},
},
});
export const {add, remove, toggle} = todoSlice.actions;
export default todoSlice.reducer;
비동기 작업
여러가지가 존재하지만, redux-thunk를 통해 만들어 보자.
redux-thunk의 경우 함수를 기반으로 작동하며 Redux Toolkit에 내장되어 있다. 나는 사실 이 방법이 리덕스에서 사용하는 가장 기본적인 비동기 처리지만 이상하게 이것만 직접 사용해 본적은 없다.
리덕스를 사용할 때는 redux-sage, redux-observable으로만 비동기 처리를 했었는데 지금은 로딩, 오류 등의 상태관리가 너무 귀찮기도 하고 리덕스의 타이핑이 줄어도 많기도 하여, react-query를 많이 사용하다 보니 더더욱 접할일이 없었다. 왜 안사용했는지도 지금은 조금 더 명확하게 알 수 있을거 같아 공부할겸 지금 해보자.
REST API 호출할 때는 많이 사용하는 Axios 라이브러리를 사용해 보겠다.
설치
$ yarn add axios
$ yarn add -D @types/axios
미들웨어 적용
기존 createStore에서 configureStore로 변경한다.
import {configureStore} from '@reduxjs/toolkit';
// ...
// const store = createStore(rootReducer);
const store = configureStore({reducer: rootReducer}); // <-- here!
function App() {
return (
<Provider store={store}>
<PostsApp />
</Provider>
);
}
export default App;
기본 사용법
createAsyncThunk 함수는 Promise를 반환하는 함수를 기반으로 함수가 호출됐을 때, 성공하거나 실패했을 때 사용할 액션들을 제공한다.
추후에 dispatch(fetchPosts())하면 상황별로 이 액션들이 dispatch된다.
import {
createAsyncThunk,
createSlice,
PayloadAction,
SerializedError,
} from '@reduxjs/toolkit';
import {getPosts} from '../api/getPosts';
import {Post} from '../api/types';
export const fetchPosts = createAsyncThunk('posts/fetchUsers', getPosts);
interface PostsState {
posts: {
loading: boolean;
data: Post[] | null;
error: SerializedError | null;
};
}
const initialState: PostsState = {
posts: {
loading: false,
data: null,
error: null,
},
};
const postsSlice = createSlice({
name: 'posts',
initialState,
reducers: {},
extraReducers: {
[fetchPosts.pending.type]: state => {
state.posts = {
loading: true,
data: null,
error: null,
};
},
[fetchPosts.fulfilled.type]: (state, action: PayloadAction<Post[]>) => {
state.posts.data = action.payload;
state.posts.loading = false;
},
[fetchPosts.rejected.type]: (
state,
action: ReturnType<typeof fetchPosts.rejected>,
) => {
state.posts.error = action.error;
state.posts.loading = false;
},
},
});
export default postsSlice.reducer;
참고로 createSlice할 때 fetchPosts를 통하여 dispatch된 액션들을 처리하는 리듀서 함수들은 extraReducers에 작성해야 한다. 왜냐하면 이 경우에 액션 생성함수와 리듀서를 동시에 만드는게 아니라 이미 정의된 액션들의 리듀서를 작성하는 것이기 때문이다.
이번에 만들 hook은 위에 훅과 달리 상태와 액션이 동시에 필요하여 한번에 만든다.
import {useCallback, useEffect} from 'react';
import {useDispatch, useSelector} from 'react-redux';
import {fetchPosts} from '../slices/posts';
interface Props {
enabled: boolean;
}
export default function usePosts({enabled}: Props) {
const posts = useSelector(state => state.posts);
const dispatch = useDispatch();
const fetchData = useCallback(() => {
dispatch(fetchPosts());
}, [dispatch]);
useEffect(() => {
if (!enabled) {
return;
}
fetchData();
}, [enabled, fetchData]);
return {
...posts,
refetch: fetchData,
};
}
Recoil
recoil에 대한 설명이 길어져 Recoil 기초 개념 및 사용법으로 대체하겠다.
useRecoilCallback(callback, deps)로 최적화 하기
먼저 기존에 작업된 코드를 보자.
import {useMemo} from 'react';
import {useRecoilValue, useSetRecoilState} from 'recoil';
import {nextTodoId, todosState} from '../atoms/todos';
export default function useTodosActions() {
const set = useSetRecoilState(todosState);
const nextId = useRecoilValue(nextTodoId);
return useMemo(
() => ({
add: (text: string) =>
set(prev => prev.concat({id: nextId.toString(), text, done: false})),
remove: (id: string) => set(prev => prev.filter(todo => todo.id !== id)),
toggle: (id: string) =>
set(prev =>
prev.map(todo =>
todo.id === id ? {...todo, done: !todo.done} : todo,
),
),
}),
[nextId, set],
);
}
useMemo가 nextId에 의존하고 있어, nextId가 바뀔때마다 useMemo에 등록한 함수가 한번 더 호출되면서 새로운 객체가 만들어 지고 있다.
사실 성능적으로는 문제가 크게 없는 부분이긴 하지만, 최적화하는 법을 배울겸 수정해 보자.
useRecoilCallback은 useCallback가 유사하지만, recoil 상태(state)에 동작하는 콜백용 api를 제공해준다.
type CallbackInterface = {
snapshot: Snapshot,
gotoSnapshot: Snapshot => void,
set: <T>(RecoilState<T>, (T => T) | T) => void,
reset: <T>(RecoilState<T>) => void,
transact_UNSTABLE: ((TransactionInterface) => void) => void,
};
function useRecoilCallback<Args, ReturnValue>(
callback: CallbackInterface => (...Args) => ReturnValue,
deps?: $ReadOnlyArray<mixed>,
): (...Args) => ReturnValue
위에서 보다시피 콜백 함수에는 여러 인터페이스가 제공되며, 위 경우 snapshot을 통해 최적화 가능하다. 스냅샷의 경우 recoil atom state에 대한 읽기 전용 보기를 제공하며, Promise형태로 반환된다.
import {useMemo} from 'react';
import {useRecoilCallback, useSetRecoilState} from 'recoil';
import {nextTodoId, todosState} from '../atoms/todos';
export default function useTodosActions() {
const set = useSetRecoilState(todosState);
const add = useRecoilCallback(
({snapshot}) =>
async (text: string) => {
const nextId = await snapshot.getPromise(nextTodoId);
set(prev => prev.concat({id: nextId.toString(), text, done: false}));
},
[],
);
return useMemo(
() => ({
add,
remove: (id: string) => set(prev => prev.filter(todo => todo.id !== id)),
toggle: (id: string) =>
set(prev =>
prev.map(todo =>
todo.id === id ? {...todo, done: !todo.done} : todo,
),
),
}),
[add, set],
);
}
이렇게 하면 nextTodoId가 바뀔 때마다 함수들이 새로 생성되지 않고 add 함수가 호출될 때 그 내부에서 현재의 nextTodoId를 조회한 뒤 해당 값을 사용해 새로운 항목을 등록하게 된다.
깃북으로 보기
'React' 카테고리의 다른 글
Recoil 기초 개념 및 사용법 (0) | 2021.12.10 |
---|---|
[React Native] 다이어리 만들기 (0) | 2021.12.07 |
[React Native] 내비게이션 및 Hooks 익히기 (0) | 2021.12.03 |
[React Native] 설치 및 할일 목록 만들기 (0) | 2021.11.29 |
React로 Threejs 예제 따라 구현하기 2 (0) | 2021.01.31 |