0
0

Delete article

Deleted articles cannot be recovered.

Draft of this article would be also deleted.

Are you sure you want to delete this article?

More than 1 year has passed since last update.

【React状態管理3】Redux

Posted at

概要

Reactの状態管理の1つである、Reduxについて学んだことをメモします。

学習内容

Reduxとは

コンポーネントツリー内でのデータ共有機能の外部パッケージの一つ。Fluxを用いている。
useContextでの状態管理で問題がある場合のみに利用するとよい。
現場で使われていることが多く、多機能のため、数ある状態管理の中でも一押し。

Redux公式サイト

Fluxパターンとは

MVCモデルの代替として用いられていた考え方。Viewから直接Actionを動かすことはせず、Dispatcherを経由して一方向に流れる。
Fluxのデザインパターン
Action -> Dispatcher -> Store -> View -> Action -> Dispatcher

  1. 画面(View)上で操作(Action)を行う
  2. ActionがDispatcherに送られる
  3. 過去の状態(State)とDispatcherを状態をReducerでまとめてStoreを更新する
    現在の状態 (prev_state) + 挙動(action) = 新しい状態(state)

Reduxの状態管理フロー

大事な3つの概念

  • 必ずグローバルStateを保存するStoreは一つにする
  • Stateは直接書き換えず、必ずactionを通す
  • reducerは純粋な関数(前回のStateとactionから新しいStateを返す)にする

Reduxの特長

  • 状態の遷移がわかりやすい
  • 長期的なメンテナンスに優れている
  • アプリの状態を一つの領域で管理できる(一元化)
  • DevToolsにより、デバッグが可能

使用方法

現在では、素のReduxを使うのではなく、Redux Toolkitを使うことが推奨されている。

手順

  1. Redux Toolkitをインストールする
    $ yarn add @reduxjs/toolkit react-redux
    
  2. createSlice()を用いて、Reduxのデータ処理を作成する(ActionやReducer、Stateといった処理を、createSliceによりまとめて書くことができる)
    import { createSlice, PayloadAction } from "@reduxjs/toolkit";
    // createSliceでは、引数にオブジェクトを持つ
    const todosSlice = createSlice({
        name: "todos", // 名称
        initialState, // 初期値
        reducers: {
            // 前回のstateとactionを受け取って新しいstateを返す処理(reducer)
        }
    });
    
  3. configureStore()を用いて、データを保持するstoreの初期値を設定する
    import { configureStore } from "@reduxjs/toolkit";
    
    export const store = configureStore({
        reducer: {
            // createSliceで設定したreducerを指定する
        },
    })
    
  4. Reduxを全ページ・コンポーネントで呼び出せるように、ソースの親ディレクトリ(_app.tsx等)にProviderでラップする(引数にstoreを指定する)
    import type { AppProps } from "next/app";
    import { Provider } from "react-redux";
    import { store } from "src/state";
    
    export default function MyApp({ Component, pageProps }: AppProps) {
        return (
            <Provider store={store}>
                <Component {...pageProps} />
            </Provider>
        );
    }
    
  5. useSelectorを用いて、ページやコンポーネントで欲しいデータだけを呼び出す
    import { FC } from "react";
    import { useSelector } from "react-redux";
    
    export const TodoCounter: FC = () => {
        const todos = useSelector((state) => state.todos);
        return (
            <h2>TODO: {todos.length}</h2>
        )
    }
    

使用例

createSliceによりReduxのデータ処理を設定する

state/todos.ts
import { createSlice, PayloadAction } from "@reduxjs/toolkit";
import { Todo } from "src/types";

const initialState: Todo[] = [
    { id: 1, text: "foo", isDone: false },
    { id: 2, text: "bar", isDone: true },
];

// createSliceを使えば、actionもreducerもひとまとめに書ける
const todosSlice = createSlice({
    name: "todos",
    initialState,
    reducers: {
        // reduxToolkitではmutableにコードを書ける(pushなどが使える)
        addTodo: (state, action: PayloadAction<Pick<Todo, "text">>) => {
            state.push({
                id: state.length + 1, 
                text: action.payload.text, 
                isDone: false,
            })
        },
        toggleTodo: (state, action: PayloadAction<Pick<Todo, "id">>) => {
            state.forEach((todo) => {
                if (todo.id === action.payload.id) {
                    todo.isDone = !todo.isDone;
                }
            })
        }
        // defaultは不要。addもtoggleでもない場合、前回のStateをそのまま返すため。
    }
});

// todosSliceを他のファイルで使えるようにexportする
export const { addTodo, toggleTodo } = todosSlice.actions; // createSliceのactionsは分割代入が使える
export const todosReducer = todosSlice.reducer;

store内のデータの初期値としてtodosReducerを設定する

state/index.ts
import { configureStore } from "@reduxjs/toolkit";
import { todosReducer } from "./todos";

export const store = configureStore({
    reducer: {
        todos: todosReducer,
    },
})

export type RootState = ReturnType<typeof store.getState>;

全てのコンポーネントをProviderでラップする

_app.tsx
import type { AppProps } from "next/app";
import { Provider } from "react-redux";
import { Layout } from "src/components/Layout";
import { store } from "src/state";

export default function MyApp({ Component, pageProps }: AppProps) {
  return (
    // Providerで子コンポーネント全体をラップする
    <Provider store={store}>
      <Layout>
        <Component {...pageProps} />
      </Layout>
    </Provider>
  );
}

コンポーネント内で欲しいデータをstoreから取得する

TodoCounter.tsx
import { FC } from "react";
import { useSelector } from "react-redux";
import { RootState } from "src/state";

export const TodoCounter: FC = () => {
    // stateからtodosを取得する
    const todos = useSelector((state: RootState) => state.todos);
    return (
        <h2>TODO: {todos.length}</h2>
    )
}

Redux DevToolsとは

GoogleChromeの拡張機能である、Redux DevToolsを用いることで、より効率的にReduxを利用できる。

Redux DevToolsの特長

  • 1つ1つのアクションを細かく見れる
  • 過去の状態を確認でき、かつ容易に戻すことができる

GoogleChromeの検証画面からアクセスする
20221024_redux_devtool1.png

使い方

action, state, diffなどでReduxの各操作の状態のチェックできる。

  • action : 処理内容
  • state : Storeの状態
  • diff : 前回処理からの差分
    20221024_redux_devtool2.png

参考

IT Kingdom
GitHubリポジトリ
Redux公式ページ
Redux Three Principles
Redux Style Guide
Redux Toolkit公式ページ

0
0
0

Register as a new user and use Qiita more conveniently

  1. You get articles that match your needs
  2. You can efficiently read back useful information
  3. You can use dark theme
What you can do with signing up
0
0

Delete article

Deleted articles cannot be recovered.

Draft of this article would be also deleted.

Are you sure you want to delete this article?