<?xml version="1.0"?>
<feed xmlns="http://www.w3.org/2005/Atom" xml:lang="ja">
	<id>http://mochiuwiki.e2.valueserver.jp/index.php?action=history&amp;feed=atom&amp;title=React%E3%81%AE%E5%9F%BA%E7%A4%8E_-_useReducer</id>
	<title>Reactの基礎 - useReducer - 版の履歴</title>
	<link rel="self" type="application/atom+xml" href="http://mochiuwiki.e2.valueserver.jp/index.php?action=history&amp;feed=atom&amp;title=React%E3%81%AE%E5%9F%BA%E7%A4%8E_-_useReducer"/>
	<link rel="alternate" type="text/html" href="http://mochiuwiki.e2.valueserver.jp/index.php?title=React%E3%81%AE%E5%9F%BA%E7%A4%8E_-_useReducer&amp;action=history"/>
	<updated>2026-07-01T06:27:17Z</updated>
	<subtitle>このウィキのこのページに関する変更履歴</subtitle>
	<generator>MediaWiki 1.43.6</generator>
	<entry>
		<id>http://mochiuwiki.e2.valueserver.jp/index.php?title=React%E3%81%AE%E5%9F%BA%E7%A4%8E_-_useReducer&amp;diff=14545&amp;oldid=prev</id>
		<title>Wiki: ページの作成:「== 概要 == &lt;code&gt;useReducer&lt;/code&gt; は、複雑な状態ロジックを管理するためのReact Hookである。&lt;br&gt; 基本構文は &lt;code&gt;const [state, dispatch] = useReducer(reducer, initialState)&lt;/code&gt; であり、現在の状態値 &lt;code&gt;state&lt;/code&gt; と、状態更新をトリガーする &lt;code&gt;dispatch&lt;/code&gt; 関数を返す。&lt;br&gt; &lt;br&gt; &lt;code&gt;reducer&lt;/code&gt; は &lt;code&gt;(state, action) =&gt; newState&lt;/code&gt; という形式の純粋関数であり、現在の状…」</title>
		<link rel="alternate" type="text/html" href="http://mochiuwiki.e2.valueserver.jp/index.php?title=React%E3%81%AE%E5%9F%BA%E7%A4%8E_-_useReducer&amp;diff=14545&amp;oldid=prev"/>
		<updated>2026-03-07T15:22:33Z</updated>

		<summary type="html">&lt;p&gt;ページの作成:「== 概要 == &amp;lt;code&amp;gt;useReducer&amp;lt;/code&amp;gt; は、複雑な状態ロジックを管理するためのReact Hookである。&amp;lt;br&amp;gt; 基本構文は &amp;lt;code&amp;gt;const [state, dispatch] = useReducer(reducer, initialState)&amp;lt;/code&amp;gt; であり、現在の状態値 &amp;lt;code&amp;gt;state&amp;lt;/code&amp;gt; と、状態更新をトリガーする &amp;lt;code&amp;gt;dispatch&amp;lt;/code&amp;gt; 関数を返す。&amp;lt;br&amp;gt; &amp;lt;br&amp;gt; &amp;lt;code&amp;gt;reducer&amp;lt;/code&amp;gt; は &amp;lt;code&amp;gt;(state, action) =&amp;gt; newState&amp;lt;/code&amp;gt; という形式の純粋関数であり、現在の状…」&lt;/p&gt;
&lt;p&gt;&lt;b&gt;新規ページ&lt;/b&gt;&lt;/p&gt;&lt;div&gt;== 概要 ==&lt;br /&gt;
&amp;lt;code&amp;gt;useReducer&amp;lt;/code&amp;gt; は、複雑な状態ロジックを管理するためのReact Hookである。&amp;lt;br&amp;gt;&lt;br /&gt;
基本構文は &amp;lt;code&amp;gt;const [state, dispatch] = useReducer(reducer, initialState)&amp;lt;/code&amp;gt; であり、現在の状態値 &amp;lt;code&amp;gt;state&amp;lt;/code&amp;gt; と、状態更新をトリガーする &amp;lt;code&amp;gt;dispatch&amp;lt;/code&amp;gt; 関数を返す。&amp;lt;br&amp;gt;&lt;br /&gt;
&amp;lt;br&amp;gt;&lt;br /&gt;
&amp;lt;code&amp;gt;reducer&amp;lt;/code&amp;gt; は &amp;lt;code&amp;gt;(state, action) =&amp;gt; newState&amp;lt;/code&amp;gt; という形式の純粋関数であり、現在の状態とアクションを受け取り、新しい状態を返す。&amp;lt;br&amp;gt;&lt;br /&gt;
この設計はReduxパターンに基づいており、状態遷移が予測可能で、テストしやすい構造になっている。&amp;lt;br&amp;gt;&lt;br /&gt;
&amp;lt;br&amp;gt;&lt;br /&gt;
&amp;lt;code&amp;gt;useReducer&amp;lt;/code&amp;gt; と &amp;lt;code&amp;gt;useState&amp;lt;/code&amp;gt; は用途によって使い分ける。&amp;lt;br&amp;gt;&lt;br /&gt;
単純な値や独立した状態には &amp;lt;code&amp;gt;useState&amp;lt;/code&amp;gt; が適しているが、複雑なオブジェクト・配列の状態管理、複数の関連する状態の一元管理、多くのアクション型が存在する場合には &amp;lt;code&amp;gt;useReducer&amp;lt;/code&amp;gt; が適している。&amp;lt;br&amp;gt;&lt;br /&gt;
目安として、&amp;lt;code&amp;gt;useState&amp;lt;/code&amp;gt; を3個以上使用している場合は &amp;lt;code&amp;gt;useReducer&amp;lt;/code&amp;gt; への移行を検討するとよい。&amp;lt;br&amp;gt;&lt;br /&gt;
&amp;lt;br&amp;gt;&lt;br /&gt;
&amp;lt;code&amp;gt;useReducer&amp;lt;/code&amp;gt; を使用することにより、状態更新ロジックをコンポーネントの外部に分離でき、コードの見通しが改善される。&amp;lt;br&amp;gt;&lt;br /&gt;
また、&amp;lt;code&amp;gt;dispatch&amp;lt;/code&amp;gt; 関数は再レンダリング間で安定した参照を持つため、&amp;lt;code&amp;gt;useCallback&amp;lt;/code&amp;gt; でラップする必要がなく、子コンポーネントへの受け渡しにも適している。&amp;lt;br&amp;gt;&lt;br /&gt;
&amp;lt;br&amp;gt;&lt;br /&gt;
TypeScriptとの相性も良く、アクション型にDiscriminated Union (判別付きユニオン) を使用することで、&amp;lt;code&amp;gt;switch&amp;lt;/code&amp;gt; 文内で型の絞り込みが自動的に行われる。&amp;lt;br&amp;gt;&lt;br /&gt;
&amp;lt;br&amp;gt;&lt;br /&gt;
第3引数として初期化関数を渡すことで、遅延初期化 (Lazy Initialization) も可能であり、初期状態の計算コストが高い場合に有用である。&amp;lt;br&amp;gt;&lt;br /&gt;
&amp;lt;br&amp;gt;&amp;lt;br&amp;gt;&lt;br /&gt;
&lt;br /&gt;
== 基本的な使用方法 ==&lt;br /&gt;
==== Reducer関数の定義 ====&lt;br /&gt;
&amp;lt;code&amp;gt;reducer&amp;lt;/code&amp;gt; 関数は、&amp;lt;code&amp;gt;(state, action) =&amp;gt; newState&amp;lt;/code&amp;gt; という形式の純粋関数として定義する。&amp;lt;br&amp;gt;&lt;br /&gt;
switch文を使用してアクションの &amp;lt;code&amp;gt;type&amp;lt;/code&amp;gt; に応じた処理に分岐し、常に新しい状態オブジェクトを返す必要がある。&amp;lt;br&amp;gt;&lt;br /&gt;
&amp;lt;br&amp;gt;&lt;br /&gt;
 &amp;lt;syntaxhighlight lang=&amp;quot;typescript&amp;quot;&amp;gt;&lt;br /&gt;
 import { useReducer } from &amp;#039;react&amp;#039;;&lt;br /&gt;
 &lt;br /&gt;
 // 状態の型定義&lt;br /&gt;
 interface CounterState {&lt;br /&gt;
    count: number;&lt;br /&gt;
 }&lt;br /&gt;
 &lt;br /&gt;
 // アクションの型定義&lt;br /&gt;
 type CounterAction =&lt;br /&gt;
    | { type: &amp;#039;INCREMENT&amp;#039; }&lt;br /&gt;
    | { type: &amp;#039;DECREMENT&amp;#039; }&lt;br /&gt;
    | { type: &amp;#039;RESET&amp;#039; }&lt;br /&gt;
    | { type: &amp;#039;SET&amp;#039;; payload: number };&lt;br /&gt;
 &lt;br /&gt;
 // Reducer関数 : 純粋関数として定義する&lt;br /&gt;
 function counterReducer(state: CounterState, action: CounterAction): CounterState {&lt;br /&gt;
    switch (action.type) {&lt;br /&gt;
       case &amp;#039;INCREMENT&amp;#039;:&lt;br /&gt;
          return { count: state.count + 1 };&lt;br /&gt;
       case &amp;#039;DECREMENT&amp;#039;:&lt;br /&gt;
          return { count: state.count - 1 };&lt;br /&gt;
       case &amp;#039;RESET&amp;#039;:&lt;br /&gt;
          return { count: 0 };&lt;br /&gt;
       case &amp;#039;SET&amp;#039;:&lt;br /&gt;
          return { count: action.payload };&lt;br /&gt;
       default:&lt;br /&gt;
          return state;&lt;br /&gt;
    }&lt;br /&gt;
 }&lt;br /&gt;
 &lt;br /&gt;
 // 初期状態&lt;br /&gt;
 const initialState: CounterState = { count: 0 };&lt;br /&gt;
 &lt;br /&gt;
 // コンポーネント&lt;br /&gt;
 function Counter() {&lt;br /&gt;
    const [state, dispatch] = useReducer(counterReducer, initialState);&lt;br /&gt;
 &lt;br /&gt;
    return (&lt;br /&gt;
       &amp;lt;div&amp;gt;&lt;br /&gt;
          &amp;lt;p&amp;gt;カウント: {state.count}&amp;lt;/p&amp;gt;&lt;br /&gt;
          &amp;lt;button onClick={() =&amp;gt; dispatch({ type: &amp;#039;INCREMENT&amp;#039; })}&amp;gt;増加&amp;lt;/button&amp;gt;&lt;br /&gt;
          &amp;lt;button onClick={() =&amp;gt; dispatch({ type: &amp;#039;DECREMENT&amp;#039; })}&amp;gt;減少&amp;lt;/button&amp;gt;&lt;br /&gt;
          &amp;lt;button onClick={() =&amp;gt; dispatch({ type: &amp;#039;RESET&amp;#039; })}&amp;gt;リセット&amp;lt;/button&amp;gt;&lt;br /&gt;
          &amp;lt;button onClick={() =&amp;gt; dispatch({ type: &amp;#039;SET&amp;#039;, payload: 10 })}&amp;gt;10に設定&amp;lt;/button&amp;gt;&lt;br /&gt;
       &amp;lt;/div&amp;gt;&lt;br /&gt;
    );&lt;br /&gt;
 }&lt;br /&gt;
 &amp;lt;/syntaxhighlight&amp;gt;&lt;br /&gt;
&amp;lt;br&amp;gt;&lt;br /&gt;
==== dispatch関数 ====&lt;br /&gt;
&amp;lt;code&amp;gt;dispatch&amp;lt;/code&amp;gt; 関数は、&amp;lt;code&amp;gt;reducer&amp;lt;/code&amp;gt; に渡すアクションオブジェクトを引数に取る。&amp;lt;br&amp;gt;&lt;br /&gt;
アクションオブジェクトには &amp;lt;code&amp;gt;type&amp;lt;/code&amp;gt; プロパティが必須であり、追加データを渡す場合は慣例として &amp;lt;code&amp;gt;payload&amp;lt;/code&amp;gt; プロパティを使用する。&amp;lt;br&amp;gt;&lt;br /&gt;
&amp;lt;br&amp;gt;&lt;br /&gt;
 &amp;lt;syntaxhighlight lang=&amp;quot;typescript&amp;quot;&amp;gt;&lt;br /&gt;
 // type のみのアクション (ペイロードなし)&lt;br /&gt;
 dispatch({ type: &amp;#039;INCREMENT&amp;#039; });&lt;br /&gt;
 dispatch({ type: &amp;#039;RESET&amp;#039; });&lt;br /&gt;
 &lt;br /&gt;
 // payload を持つアクション&lt;br /&gt;
 dispatch({ type: &amp;#039;SET&amp;#039;, payload: 42 });&lt;br /&gt;
 dispatch({ type: &amp;#039;SET_NAME&amp;#039;, payload: &amp;#039;Alice&amp;#039; });&lt;br /&gt;
 &amp;lt;/syntaxhighlight&amp;gt;&lt;br /&gt;
&amp;lt;br&amp;gt;&lt;br /&gt;
&amp;lt;code&amp;gt;dispatch&amp;lt;/code&amp;gt; は同期的に処理されるため、&amp;lt;code&amp;gt;dispatch&amp;lt;/code&amp;gt; 呼び出し直後に &amp;lt;code&amp;gt;state&amp;lt;/code&amp;gt; を参照しても、更新後の値は得られない点に注意する。&amp;lt;br&amp;gt;&lt;br /&gt;
更新後の状態が反映されるのは次のレンダリング以降である。&amp;lt;br&amp;gt;&lt;br /&gt;
&amp;lt;br&amp;gt;&amp;lt;br&amp;gt;&lt;br /&gt;
&lt;br /&gt;
== TypeScriptでの型定義 ==&lt;br /&gt;
==== Discriminated Union型 ====&lt;br /&gt;
TypeScriptでは、アクション型をDiscriminated Union (判別可能なユニオン型) として定義することを推奨する。&amp;lt;br&amp;gt;&lt;br /&gt;
各アクションオブジェクトの &amp;lt;code&amp;gt;type&amp;lt;/code&amp;gt; プロパティをリテラル型にすることで、TypeScriptが各caseブロック内でアクションの型を自動的に絞り込む。&amp;lt;br&amp;gt;&lt;br /&gt;
&amp;lt;br&amp;gt;&lt;br /&gt;
 &amp;lt;syntaxhighlight lang=&amp;quot;typescript&amp;quot;&amp;gt;&lt;br /&gt;
 // Discriminated Union型によるアクション定義&lt;br /&gt;
 type Action =&lt;br /&gt;
    | { type: &amp;#039;INCREMENT&amp;#039; }&lt;br /&gt;
    | { type: &amp;#039;DECREMENT&amp;#039; }&lt;br /&gt;
    | { type: &amp;#039;SET_COUNT&amp;#039;; payload: number }&lt;br /&gt;
    | { type: &amp;#039;SET_NAME&amp;#039;; payload: string }&lt;br /&gt;
    | { type: &amp;#039;RESET&amp;#039; };&lt;br /&gt;
 &lt;br /&gt;
 function reducer(state: State, action: Action): State {&lt;br /&gt;
    switch (action.type) {&lt;br /&gt;
       case &amp;#039;SET_COUNT&amp;#039;:&lt;br /&gt;
          // この case ブロック内では action.payload が number 型に絞り込まれる&lt;br /&gt;
          return { ...state, count: action.payload };&lt;br /&gt;
       case &amp;#039;SET_NAME&amp;#039;:&lt;br /&gt;
          // この case ブロック内では action.payload が string 型に絞り込まれる&lt;br /&gt;
          return { ...state, name: action.payload };&lt;br /&gt;
       case &amp;#039;INCREMENT&amp;#039;:&lt;br /&gt;
          // この case ブロック内では action に payload プロパティは存在しない&lt;br /&gt;
          return { ...state, count: state.count + 1 };&lt;br /&gt;
       default:&lt;br /&gt;
          return state;&lt;br /&gt;
    }&lt;br /&gt;
 }&lt;br /&gt;
 &amp;lt;/syntaxhighlight&amp;gt;&lt;br /&gt;
&amp;lt;br&amp;gt;&lt;br /&gt;
==== State型の定義 ====&lt;br /&gt;
状態の型は &amp;lt;code&amp;gt;interface&amp;lt;/code&amp;gt; または &amp;lt;code&amp;gt;type&amp;lt;/code&amp;gt; を使用して定義する。&amp;lt;br&amp;gt;&lt;br /&gt;
複雑な状態を持つ場合でも、型定義により状態の構造が明確になる。&amp;lt;br&amp;gt;&lt;br /&gt;
&amp;lt;br&amp;gt;&lt;br /&gt;
 &amp;lt;syntaxhighlight lang=&amp;quot;typescript&amp;quot;&amp;gt;&lt;br /&gt;
 // State型の定義&lt;br /&gt;
 interface State {&lt;br /&gt;
    count: number;&lt;br /&gt;
    history: number[];&lt;br /&gt;
    name: string;&lt;br /&gt;
    isLoading: boolean;&lt;br /&gt;
    error: string | null;&lt;br /&gt;
 }&lt;br /&gt;
 &lt;br /&gt;
 // 初期状態&lt;br /&gt;
 const initialState: State = {&lt;br /&gt;
    count: 0,&lt;br /&gt;
    history: [],&lt;br /&gt;
    name: &amp;#039;&amp;#039;,&lt;br /&gt;
    isLoading: false,&lt;br /&gt;
    error: null,&lt;br /&gt;
 };&lt;br /&gt;
 &lt;br /&gt;
 // アクション型の定義&lt;br /&gt;
 type Action =&lt;br /&gt;
    | { type: &amp;#039;INCREMENT&amp;#039; }&lt;br /&gt;
    | { type: &amp;#039;DECREMENT&amp;#039; }&lt;br /&gt;
    | { type: &amp;#039;SET_COUNT&amp;#039;; payload: number }&lt;br /&gt;
    | { type: &amp;#039;SET_NAME&amp;#039;; payload: string }&lt;br /&gt;
    | { type: &amp;#039;SET_LOADING&amp;#039;; payload: boolean }&lt;br /&gt;
    | { type: &amp;#039;SET_ERROR&amp;#039;; payload: string | null }&lt;br /&gt;
    | { type: &amp;#039;RESET&amp;#039; };&lt;br /&gt;
 &lt;br /&gt;
 function reducer(state: State, action: Action): State {&lt;br /&gt;
    switch (action.type) {&lt;br /&gt;
       case &amp;#039;INCREMENT&amp;#039;:&lt;br /&gt;
          return {&lt;br /&gt;
             ...state,&lt;br /&gt;
             count: state.count + 1,&lt;br /&gt;
             history: [...state.history, state.count + 1],&lt;br /&gt;
          };&lt;br /&gt;
       case &amp;#039;DECREMENT&amp;#039;:&lt;br /&gt;
          return {&lt;br /&gt;
             ...state,&lt;br /&gt;
             count: state.count - 1,&lt;br /&gt;
             history: [...state.history, state.count - 1],&lt;br /&gt;
          };&lt;br /&gt;
       case &amp;#039;SET_COUNT&amp;#039;:&lt;br /&gt;
          return { ...state, count: action.payload };&lt;br /&gt;
       case &amp;#039;SET_NAME&amp;#039;:&lt;br /&gt;
          return { ...state, name: action.payload };&lt;br /&gt;
       case &amp;#039;SET_LOADING&amp;#039;:&lt;br /&gt;
          return { ...state, isLoading: action.payload };&lt;br /&gt;
       case &amp;#039;SET_ERROR&amp;#039;:&lt;br /&gt;
          return { ...state, error: action.payload };&lt;br /&gt;
       case &amp;#039;RESET&amp;#039;:&lt;br /&gt;
          return initialState;&lt;br /&gt;
       default:&lt;br /&gt;
          return state;&lt;br /&gt;
    }&lt;br /&gt;
 }&lt;br /&gt;
 &amp;lt;/syntaxhighlight&amp;gt;&lt;br /&gt;
&amp;lt;br&amp;gt;&amp;lt;br&amp;gt;&lt;br /&gt;
&lt;br /&gt;
== useReducerとuseStateの使い分け ==&lt;br /&gt;
&amp;lt;code&amp;gt;useReducer&amp;lt;/code&amp;gt; と &amp;lt;code&amp;gt;useState&amp;lt;/code&amp;gt; はどちらも状態管理に使用するが、それぞれに適したユースケースが異なる。&amp;lt;br&amp;gt;&lt;br /&gt;
&amp;lt;br&amp;gt;&lt;br /&gt;
&amp;lt;center&amp;gt;&lt;br /&gt;
{| class=&amp;quot;wikitable&amp;quot;&lt;br /&gt;
|+ useReducer と useStateの比較&lt;br /&gt;
! 比較項目 !! useState !! useReducer&lt;br /&gt;
|-&lt;br /&gt;
| 状態の複雑さ || 単純な値 (数値・文字列・真偽値) || 複雑なオブジェクトや配列&lt;br /&gt;
|-&lt;br /&gt;
| 状態の構造 || 独立した単一の値 || 複数の関連した値をまとめた構造体&lt;br /&gt;
|-&lt;br /&gt;
| アクション数 || 1〜2種類の更新操作 || 多くの種類のアクションが存在する。&lt;br /&gt;
|-&lt;br /&gt;
| 状態の依存性 || 各状態が独立している。 || 次の状態が現在の状態に依存する。&lt;br /&gt;
|-&lt;br /&gt;
| テスト容易性 || コンポーネントと一体 || reducerを単体でテストできる。&lt;br /&gt;
|-&lt;br /&gt;
| ロジックの分離 || コンポーネント内に記述 || コンポーネント外に分離できる。&lt;br /&gt;
|-&lt;br /&gt;
| 推奨規模 || 小規模・単純なコンポーネント || 中〜大規模・複雑な状態管理&lt;br /&gt;
|}&lt;br /&gt;
&amp;lt;/center&amp;gt;&lt;br /&gt;
&amp;lt;br&amp;gt;&lt;br /&gt;
下表に、使い分けの判断基準を示す。&amp;lt;br&amp;gt;&lt;br /&gt;
&amp;lt;br&amp;gt;&lt;br /&gt;
&amp;lt;center&amp;gt;&lt;br /&gt;
{| class=&amp;quot;wikitable&amp;quot;&lt;br /&gt;
|+ 使い分けの判断基準&lt;br /&gt;
! 状況 !! 推奨するHook&lt;br /&gt;
|-&lt;br /&gt;
| カウンターや入力値など単一の値を管理する場合 || useState&lt;br /&gt;
|-&lt;br /&gt;
| 真偽値のトグル操作 || useState&lt;br /&gt;
|-&lt;br /&gt;
| 3つ以上の useState が存在し、関連している場合 || useReducer&lt;br /&gt;
|-&lt;br /&gt;
| 状態の更新に複数のアクションパターンがある場合 || useReducer&lt;br /&gt;
|-&lt;br /&gt;
| 次の状態が現在の状態の複数フィールドに依存する場合 || useReducer&lt;br /&gt;
|-&lt;br /&gt;
| 状態更新ロジックをコンポーネント外にテストしたい場合 || useReducer&lt;br /&gt;
|}&lt;br /&gt;
&amp;lt;/center&amp;gt;&lt;br /&gt;
&amp;lt;br&amp;gt;&amp;lt;br&amp;gt;&lt;br /&gt;
&lt;br /&gt;
== 遅延初期化 ==&lt;br /&gt;
&amp;lt;code&amp;gt;useReducer&amp;lt;/code&amp;gt; の第3引数に初期化関数 (&amp;lt;code&amp;gt;init&amp;lt;/code&amp;gt;) を渡すことにより、遅延初期化を実現できる。&amp;lt;br&amp;gt;&lt;br /&gt;
初期化関数は初回レンダリング時にのみ実行されるため、処理コストの高い初期状態の計算を遅延させるのに適している。&amp;lt;br&amp;gt;&lt;br /&gt;
&amp;lt;br&amp;gt;&lt;br /&gt;
構文は &amp;lt;code&amp;gt;useReducer(reducer, initialArg, init)&amp;lt;/code&amp;gt; であり、&amp;lt;code&amp;gt;init(initialArg)&amp;lt;/code&amp;gt; の戻り値が初期状態として使用される。&amp;lt;br&amp;gt;&lt;br /&gt;
&amp;lt;br&amp;gt;&lt;br /&gt;
 &amp;lt;syntaxhighlight lang=&amp;quot;typescript&amp;quot;&amp;gt;&lt;br /&gt;
 interface State {&lt;br /&gt;
    count: number;&lt;br /&gt;
    items: string[];&lt;br /&gt;
 }&lt;br /&gt;
 &lt;br /&gt;
 type Action =&lt;br /&gt;
    | { type: &amp;#039;INCREMENT&amp;#039; }&lt;br /&gt;
    | { type: &amp;#039;RESET&amp;#039;; payload: string }&lt;br /&gt;
    | { type: &amp;#039;ADD_ITEM&amp;#039;; payload: string };&lt;br /&gt;
 &lt;br /&gt;
 // 初期化関数 : 引数を受け取り、初期状態を返す&lt;br /&gt;
 // この関数は初回レンダリング時のみ実行される&lt;br /&gt;
 function createInitialState(userId: string): State {&lt;br /&gt;
    // 処理コストの高い初期化処理&lt;br /&gt;
    const savedData = localStorage.getItem(`state_${userId}`);&lt;br /&gt;
    if (savedData) {&lt;br /&gt;
       return JSON.parse(savedData) as State;&lt;br /&gt;
    }&lt;br /&gt;
    return { count: 0, items: [] };&lt;br /&gt;
 }&lt;br /&gt;
 &lt;br /&gt;
 function reducer(state: State, action: Action): State {&lt;br /&gt;
    switch (action.type) {&lt;br /&gt;
       case &amp;#039;INCREMENT&amp;#039;:&lt;br /&gt;
          return { ...state, count: state.count + 1 };&lt;br /&gt;
       case &amp;#039;RESET&amp;#039;:&lt;br /&gt;
          // init関数を使用してリセット&lt;br /&gt;
          return createInitialState(action.payload);&lt;br /&gt;
       case &amp;#039;ADD_ITEM&amp;#039;:&lt;br /&gt;
          return { ...state, items: [...state.items, action.payload] };&lt;br /&gt;
       default:&lt;br /&gt;
          return state;&lt;br /&gt;
    }&lt;br /&gt;
 }&lt;br /&gt;
 &lt;br /&gt;
 function UserCounter({ userId }: { userId: string }) {&lt;br /&gt;
    // 第3引数にinit関数を指定 : createInitialState(userId) が初回のみ実行される&lt;br /&gt;
    const [state, dispatch] = useReducer(reducer, userId, createInitialState);&lt;br /&gt;
 &lt;br /&gt;
    return (&lt;br /&gt;
       &amp;lt;div&amp;gt;&lt;br /&gt;
          &amp;lt;p&amp;gt;カウント: {state.count}&amp;lt;/p&amp;gt;&lt;br /&gt;
          &amp;lt;button onClick={() =&amp;gt; dispatch({ type: &amp;#039;INCREMENT&amp;#039; })}&amp;gt;増加&amp;lt;/button&amp;gt;&lt;br /&gt;
          &amp;lt;button onClick={() =&amp;gt; dispatch({ type: &amp;#039;RESET&amp;#039;, payload: userId })}&amp;gt;リセット&amp;lt;/button&amp;gt;&lt;br /&gt;
       &amp;lt;/div&amp;gt;&lt;br /&gt;
    );&lt;br /&gt;
 }&lt;br /&gt;
 &amp;lt;/syntaxhighlight&amp;gt;&lt;br /&gt;
&amp;lt;br&amp;gt;&lt;br /&gt;
遅延初期化には、&amp;lt;code&amp;gt;reset&amp;lt;/code&amp;gt; アクション時に &amp;lt;code&amp;gt;init&amp;lt;/code&amp;gt; 関数を再利用して初期状態に戻すことができるメリットもある。&amp;lt;br&amp;gt;&lt;br /&gt;
&amp;lt;br&amp;gt;&amp;lt;br&amp;gt;&lt;br /&gt;
&lt;br /&gt;
== サンプルコード : Reducerパターン ==&lt;br /&gt;
==== Todoリストの管理 ====&lt;br /&gt;
複数のアクション型を持つTodoリストは、&amp;lt;code&amp;gt;useReducer&amp;lt;/code&amp;gt; が効果的に機能するユースケースの代表例である。&amp;lt;br&amp;gt;&lt;br /&gt;
&amp;lt;br&amp;gt;&lt;br /&gt;
 &amp;lt;syntaxhighlight lang=&amp;quot;typescript&amp;quot;&amp;gt;&lt;br /&gt;
 interface Todo {&lt;br /&gt;
    id: number;&lt;br /&gt;
    text: string;&lt;br /&gt;
    completed: boolean;&lt;br /&gt;
 }&lt;br /&gt;
 &lt;br /&gt;
 type FilterType = &amp;#039;ALL&amp;#039; | &amp;#039;ACTIVE&amp;#039; | &amp;#039;COMPLETED&amp;#039;;&lt;br /&gt;
 &lt;br /&gt;
 interface TodoState {&lt;br /&gt;
    todos: Todo[];&lt;br /&gt;
    filter: FilterType;&lt;br /&gt;
    nextId: number;&lt;br /&gt;
 }&lt;br /&gt;
 &lt;br /&gt;
 type TodoAction =&lt;br /&gt;
    | { type: &amp;#039;ADD&amp;#039;; payload: string }&lt;br /&gt;
    | { type: &amp;#039;REMOVE&amp;#039;; payload: number }&lt;br /&gt;
    | { type: &amp;#039;TOGGLE&amp;#039;; payload: number }&lt;br /&gt;
    | { type: &amp;#039;SET_FILTER&amp;#039;; payload: FilterType }&lt;br /&gt;
    | { type: &amp;#039;CLEAR_COMPLETED&amp;#039; };&lt;br /&gt;
 &lt;br /&gt;
 const initialTodoState: TodoState = {&lt;br /&gt;
    todos: [],&lt;br /&gt;
    filter: &amp;#039;ALL&amp;#039;,&lt;br /&gt;
    nextId: 1,&lt;br /&gt;
 };&lt;br /&gt;
 &lt;br /&gt;
 function todoReducer(state: TodoState, action: TodoAction): TodoState {&lt;br /&gt;
    switch (action.type) {&lt;br /&gt;
       case &amp;#039;ADD&amp;#039;:&lt;br /&gt;
          return {&lt;br /&gt;
             ...state,&lt;br /&gt;
             todos: [&lt;br /&gt;
                ...state.todos,&lt;br /&gt;
                { id: state.nextId, text: action.payload, completed: false },&lt;br /&gt;
             ],&lt;br /&gt;
             nextId: state.nextId + 1,&lt;br /&gt;
          };&lt;br /&gt;
       case &amp;#039;REMOVE&amp;#039;:&lt;br /&gt;
          return {&lt;br /&gt;
             ...state,&lt;br /&gt;
             todos: state.todos.filter((todo) =&amp;gt; todo.id !== action.payload),&lt;br /&gt;
          };&lt;br /&gt;
       case &amp;#039;TOGGLE&amp;#039;:&lt;br /&gt;
          return {&lt;br /&gt;
             ...state,&lt;br /&gt;
             todos: state.todos.map((todo) =&amp;gt;&lt;br /&gt;
                todo.id === action.payload&lt;br /&gt;
                   ? { ...todo, completed: !todo.completed }&lt;br /&gt;
                   : todo&lt;br /&gt;
             ),&lt;br /&gt;
          };&lt;br /&gt;
       case &amp;#039;SET_FILTER&amp;#039;:&lt;br /&gt;
          return { ...state, filter: action.payload };&lt;br /&gt;
       case &amp;#039;CLEAR_COMPLETED&amp;#039;:&lt;br /&gt;
          return {&lt;br /&gt;
             ...state,&lt;br /&gt;
             todos: state.todos.filter((todo) =&amp;gt; !todo.completed),&lt;br /&gt;
          };&lt;br /&gt;
       default:&lt;br /&gt;
          return state;&lt;br /&gt;
    }&lt;br /&gt;
 }&lt;br /&gt;
 &lt;br /&gt;
 function TodoApp() {&lt;br /&gt;
    const [state, dispatch] = useReducer(todoReducer, initialTodoState);&lt;br /&gt;
    const [inputText, setInputText] = useState(&amp;#039;&amp;#039;);&lt;br /&gt;
 &lt;br /&gt;
    const filteredTodos = state.todos.filter((todo) =&amp;gt; {&lt;br /&gt;
       if (state.filter === &amp;#039;ACTIVE&amp;#039;) return !todo.completed;&lt;br /&gt;
       if (state.filter === &amp;#039;COMPLETED&amp;#039;) return todo.completed;&lt;br /&gt;
       return true;&lt;br /&gt;
    });&lt;br /&gt;
&lt;br /&gt;
    return (&lt;br /&gt;
       &amp;lt;div&amp;gt;&lt;br /&gt;
          &amp;lt;input&lt;br /&gt;
             value={inputText}&lt;br /&gt;
             onChange={(e) =&amp;gt; setInputText(e.target.value)}&lt;br /&gt;
             placeholder=&amp;quot;新しいTodoを入力&amp;quot;&lt;br /&gt;
          /&amp;gt;&lt;br /&gt;
          &amp;lt;button onClick={() =&amp;gt; {&lt;br /&gt;
             if (inputText.trim()) {&lt;br /&gt;
                dispatch({ type: &amp;#039;ADD&amp;#039;, payload: inputText.trim() });&lt;br /&gt;
                setInputText(&amp;#039;&amp;#039;);&lt;br /&gt;
             }&lt;br /&gt;
          }}&amp;gt;追加&amp;lt;/button&amp;gt;&lt;br /&gt;
          &amp;lt;div&amp;gt;&lt;br /&gt;
             &amp;lt;button onClick={() =&amp;gt; dispatch({ type: &amp;#039;SET_FILTER&amp;#039;, payload: &amp;#039;ALL&amp;#039; })}&amp;gt;全て&amp;lt;/button&amp;gt;&lt;br /&gt;
             &amp;lt;button onClick={() =&amp;gt; dispatch({ type: &amp;#039;SET_FILTER&amp;#039;, payload: &amp;#039;ACTIVE&amp;#039; })}&amp;gt;未完了&amp;lt;/button&amp;gt;&lt;br /&gt;
             &amp;lt;button onClick={() =&amp;gt; dispatch({ type: &amp;#039;SET_FILTER&amp;#039;, payload: &amp;#039;COMPLETED&amp;#039; })}&amp;gt;完了&amp;lt;/button&amp;gt;&lt;br /&gt;
          &amp;lt;/div&amp;gt;&lt;br /&gt;
          &amp;lt;ul&amp;gt;&lt;br /&gt;
             {filteredTodos.map((todo) =&amp;gt; (&lt;br /&gt;
                &amp;lt;li key={todo.id}&amp;gt;&lt;br /&gt;
                   &amp;lt;span&lt;br /&gt;
                      style={{ textDecoration: todo.completed ? &amp;#039;line-through&amp;#039; : &amp;#039;none&amp;#039; }}&lt;br /&gt;
                      onClick={() =&amp;gt; dispatch({ type: &amp;#039;TOGGLE&amp;#039;, payload: todo.id })}&lt;br /&gt;
                   &amp;gt;&lt;br /&gt;
                      {todo.text}&lt;br /&gt;
                   &amp;lt;/span&amp;gt;&lt;br /&gt;
                   &amp;lt;button onClick={() =&amp;gt; dispatch({ type: &amp;#039;REMOVE&amp;#039;, payload: todo.id })}&amp;gt;削除&amp;lt;/button&amp;gt;&lt;br /&gt;
                &amp;lt;/li&amp;gt;&lt;br /&gt;
             ))}&lt;br /&gt;
          &amp;lt;/ul&amp;gt;&lt;br /&gt;
          &amp;lt;button onClick={() =&amp;gt; dispatch({ type: &amp;#039;CLEAR_COMPLETED&amp;#039; })}&amp;gt;完了済みを削除&amp;lt;/button&amp;gt;&lt;br /&gt;
       &amp;lt;/div&amp;gt;&lt;br /&gt;
    );&lt;br /&gt;
 }&lt;br /&gt;
 &amp;lt;/syntaxhighlight&amp;gt;&lt;br /&gt;
&amp;lt;br&amp;gt;&lt;br /&gt;
==== フォーム状態の管理 ====&lt;br /&gt;
複数の入力フィールドを持つフォームの状態を、&amp;lt;code&amp;gt;useReducer&amp;lt;/code&amp;gt; で一元管理する例を以下に示す。&amp;lt;br&amp;gt;&lt;br /&gt;
&amp;lt;br&amp;gt;&lt;br /&gt;
 &amp;lt;syntaxhighlight lang=&amp;quot;typescript&amp;quot;&amp;gt;&lt;br /&gt;
 interface FormState {&lt;br /&gt;
    name: string;&lt;br /&gt;
    email: string;&lt;br /&gt;
    password: string;&lt;br /&gt;
    isSubmitting: boolean;&lt;br /&gt;
    errors: Record&amp;lt;string, string&amp;gt;;&lt;br /&gt;
 }&lt;br /&gt;
 &lt;br /&gt;
 type FormAction =&lt;br /&gt;
    | { type: &amp;#039;SET_FIELD&amp;#039;; field: keyof Pick&amp;lt;FormState, &amp;#039;name&amp;#039; | &amp;#039;email&amp;#039; | &amp;#039;password&amp;#039;&amp;gt;; payload: string }&lt;br /&gt;
    | { type: &amp;#039;SET_SUBMITTING&amp;#039;; payload: boolean }&lt;br /&gt;
    | { type: &amp;#039;SET_ERROR&amp;#039;; field: string; payload: string }&lt;br /&gt;
    | { type: &amp;#039;CLEAR_ERRORS&amp;#039; }&lt;br /&gt;
    | { type: &amp;#039;RESET&amp;#039; };&lt;br /&gt;
 &lt;br /&gt;
 const initialFormState: FormState = {&lt;br /&gt;
    name: &amp;#039;&amp;#039;,&lt;br /&gt;
    email: &amp;#039;&amp;#039;,&lt;br /&gt;
    password: &amp;#039;&amp;#039;,&lt;br /&gt;
    isSubmitting: false,&lt;br /&gt;
    errors: {},&lt;br /&gt;
 };&lt;br /&gt;
 &lt;br /&gt;
 function formReducer(state: FormState, action: FormAction): FormState {&lt;br /&gt;
    switch (action.type) {&lt;br /&gt;
       case &amp;#039;SET_FIELD&amp;#039;:&lt;br /&gt;
          return { ...state, [action.field]: action.payload };&lt;br /&gt;
       case &amp;#039;SET_SUBMITTING&amp;#039;:&lt;br /&gt;
          return { ...state, isSubmitting: action.payload };&lt;br /&gt;
       case &amp;#039;SET_ERROR&amp;#039;:&lt;br /&gt;
          return {&lt;br /&gt;
             ...state,&lt;br /&gt;
             errors: { ...state.errors, [action.field]: action.payload },&lt;br /&gt;
          };&lt;br /&gt;
       case &amp;#039;CLEAR_ERRORS&amp;#039;:&lt;br /&gt;
          return { ...state, errors: {} };&lt;br /&gt;
       case &amp;#039;RESET&amp;#039;:&lt;br /&gt;
          return initialFormState;&lt;br /&gt;
       default:&lt;br /&gt;
          return state;&lt;br /&gt;
    }&lt;br /&gt;
 }&lt;br /&gt;
 &lt;br /&gt;
 function RegistrationForm() {&lt;br /&gt;
    const [state, dispatch] = useReducer(formReducer, initialFormState);&lt;br /&gt;
&lt;br /&gt;
    const handleSubmit = async (e: React.FormEvent) =&amp;gt; {&lt;br /&gt;
       e.preventDefault();&lt;br /&gt;
       dispatch({ type: &amp;#039;CLEAR_ERRORS&amp;#039; });&lt;br /&gt;
       dispatch({ type: &amp;#039;SET_SUBMITTING&amp;#039;, payload: true });&lt;br /&gt;
 &lt;br /&gt;
       try {&lt;br /&gt;
          // フォーム送信処理&lt;br /&gt;
          await submitForm(state);&lt;br /&gt;
       }&lt;br /&gt;
       catch (error) {&lt;br /&gt;
          dispatch({ type: &amp;#039;SET_ERROR&amp;#039;, field: &amp;#039;submit&amp;#039;, payload: &amp;#039;送信に失敗しました&amp;#039; });&lt;br /&gt;
       }&lt;br /&gt;
       finally {&lt;br /&gt;
          dispatch({ type: &amp;#039;SET_SUBMITTING&amp;#039;, payload: false });&lt;br /&gt;
       }&lt;br /&gt;
    };&lt;br /&gt;
 &lt;br /&gt;
    return (&lt;br /&gt;
       &amp;lt;form onSubmit={handleSubmit}&amp;gt;&lt;br /&gt;
          &amp;lt;input&lt;br /&gt;
             value={state.name}&lt;br /&gt;
             onChange={(e) =&amp;gt; dispatch({ type: &amp;#039;SET_FIELD&amp;#039;, field: &amp;#039;name&amp;#039;, payload: e.target.value })}&lt;br /&gt;
             placeholder=&amp;quot;名前&amp;quot;&lt;br /&gt;
          /&amp;gt;&lt;br /&gt;
          &amp;lt;input&lt;br /&gt;
             value={state.email}&lt;br /&gt;
             onChange={(e) =&amp;gt; dispatch({ type: &amp;#039;SET_FIELD&amp;#039;, field: &amp;#039;email&amp;#039;, payload: e.target.value })}&lt;br /&gt;
             placeholder=&amp;quot;メールアドレス&amp;quot;&lt;br /&gt;
          /&amp;gt;&lt;br /&gt;
          &amp;lt;input&lt;br /&gt;
             type=&amp;quot;password&amp;quot;&lt;br /&gt;
             value={state.password}&lt;br /&gt;
             onChange={(e) =&amp;gt; dispatch({ type: &amp;#039;SET_FIELD&amp;#039;, field: &amp;#039;password&amp;#039;, payload: e.target.value })}&lt;br /&gt;
             placeholder=&amp;quot;パスワード&amp;quot;&lt;br /&gt;
          /&amp;gt;&lt;br /&gt;
          {state.errors.submit &amp;amp;&amp;amp; &amp;lt;p&amp;gt;{state.errors.submit}&amp;lt;/p&amp;gt;}&lt;br /&gt;
          &amp;lt;button type=&amp;quot;submit&amp;quot; disabled={state.isSubmitting}&amp;gt;&lt;br /&gt;
             {state.isSubmitting ? &amp;#039;送信中...&amp;#039; : &amp;#039;登録&amp;#039;}&lt;br /&gt;
          &amp;lt;/button&amp;gt;&lt;br /&gt;
       &amp;lt;/form&amp;gt;&lt;br /&gt;
    );&lt;br /&gt;
 }&lt;br /&gt;
 &amp;lt;/syntaxhighlight&amp;gt;&lt;br /&gt;
&amp;lt;br&amp;gt;&amp;lt;br&amp;gt;&lt;br /&gt;
&lt;br /&gt;
== Reduxパターンとの比較 ==&lt;br /&gt;
&amp;lt;code&amp;gt;useReducer&amp;lt;/code&amp;gt; はReduxと同じ設計思想に基づいているが、スコープや機能面で重要な違いがある。&amp;lt;br&amp;gt;&lt;br /&gt;
&amp;lt;br&amp;gt;&lt;br /&gt;
&amp;lt;center&amp;gt;&lt;br /&gt;
{| class=&amp;quot;wikitable&amp;quot;&lt;br /&gt;
|+ useReducerとReduxの比較&lt;br /&gt;
! 比較項目 !! useReducer !! Redux (Redux Toolkit)&lt;br /&gt;
|-&lt;br /&gt;
| スコープ || 単一コンポーネント (またはContextを通じたサブツリー) || アプリケーション全体&lt;br /&gt;
|-&lt;br /&gt;
| セットアップ || React組み込みのため追加インストール不要 || パッケージのインストールが必要&lt;br /&gt;
|-&lt;br /&gt;
| ミドルウェア || 非対応 || redux-thunk、redux-saga等を使用可能&lt;br /&gt;
|-&lt;br /&gt;
| DevTools || 非対応 || Redux DevToolsで状態遷移を可視化できる。&lt;br /&gt;
|-&lt;br /&gt;
| 非同期処理 || useEffectと組み合わせる必要がある。 || createAsyncThunk等で統合的に処理できる。&lt;br /&gt;
|-&lt;br /&gt;
| 拡張性 || コンポーネントレベルに限定される || プラグインやミドルウェアで拡張できる。&lt;br /&gt;
|-&lt;br /&gt;
| 学習コスト || 低い (React Hooksの知識のみ) || 高い (Redux固有の概念が多い)&lt;br /&gt;
|-&lt;br /&gt;
| 適した規模 || 小〜中規模アプリケーション || 中〜大規模アプリケーション&lt;br /&gt;
|}&lt;br /&gt;
&amp;lt;/center&amp;gt;&lt;br /&gt;
&amp;lt;br&amp;gt;&lt;br /&gt;
下表に、類似点をまとめる。&amp;lt;br&amp;gt;&lt;br /&gt;
&amp;lt;br&amp;gt;&lt;br /&gt;
&amp;lt;center&amp;gt;&lt;br /&gt;
{| class=&amp;quot;wikitable&amp;quot;&lt;br /&gt;
|+ useReducer と Reduxの類似点&lt;br /&gt;
! 共通概念 !! 説明&lt;br /&gt;
|-&lt;br /&gt;
| reducer関数 || どちらも &amp;lt;code&amp;gt;(state, action) =&amp;gt; newState&amp;lt;/code&amp;gt; の純粋関数を使用する。&lt;br /&gt;
|-&lt;br /&gt;
| アクションオブジェクト || どちらも &amp;lt;code&amp;gt;type&amp;lt;/code&amp;gt; プロパティを持つアクションオブジェクトを使用する。&lt;br /&gt;
|-&lt;br /&gt;
| 予測可能な状態遷移 || 同じアクションと状態に対して常に同じ結果を返す。&lt;br /&gt;
|-&lt;br /&gt;
| イミュータブルな状態更新 || 既存の状態を変更せず、新しい状態オブジェクトを返す。&lt;br /&gt;
|}&lt;br /&gt;
&amp;lt;/center&amp;gt;&lt;br /&gt;
&amp;lt;br&amp;gt;&amp;lt;br&amp;gt;&lt;br /&gt;
&lt;br /&gt;
== useReducer と useContextの組み合わせ ==&lt;br /&gt;
&amp;lt;code&amp;gt;useReducer&amp;lt;/code&amp;gt; と &amp;lt;code&amp;gt;useContext&amp;lt;/code&amp;gt; を組み合わせることにより、コンポーネントツリー全体でグローバルな状態管理を実現できる。&amp;lt;br&amp;gt;&lt;br /&gt;
Props drilling (プロップスのバケツリレー) を避けながら、複数のコンポーネントから状態にアクセスできるようになる。&amp;lt;br&amp;gt;&lt;br /&gt;
&amp;lt;br&amp;gt;&lt;br /&gt;
状態と &amp;lt;code&amp;gt;dispatch&amp;lt;/code&amp;gt; を分離してContextを定義することで、&amp;lt;code&amp;gt;dispatch&amp;lt;/code&amp;gt; のみを使用するコンポーネントの不要な再レンダリングを防ぐことができる。&amp;lt;br&amp;gt;&lt;br /&gt;
&amp;lt;br&amp;gt;&lt;br /&gt;
 &amp;lt;syntaxhighlight lang=&amp;quot;typescript&amp;quot;&amp;gt;&lt;br /&gt;
 import { createContext, useContext, useReducer, ReactNode } from &amp;#039;react&amp;#039;;&lt;br /&gt;
 &lt;br /&gt;
 // 型定義&lt;br /&gt;
 interface Task {&lt;br /&gt;
    id: number;&lt;br /&gt;
    text: string;&lt;br /&gt;
    completed: boolean;&lt;br /&gt;
 }&lt;br /&gt;
 &lt;br /&gt;
 interface TaskState {&lt;br /&gt;
    tasks: Task[];&lt;br /&gt;
    nextId: number;&lt;br /&gt;
 }&lt;br /&gt;
 &lt;br /&gt;
 type TaskAction =&lt;br /&gt;
    | { type: &amp;#039;ADD&amp;#039;; payload: string }&lt;br /&gt;
    | { type: &amp;#039;REMOVE&amp;#039;; payload: number }&lt;br /&gt;
    | { type: &amp;#039;TOGGLE&amp;#039;; payload: number };&lt;br /&gt;
 &lt;br /&gt;
 // Contextの型定義 (StateとDispatchを分離)&lt;br /&gt;
 const TaskStateContext = createContext&amp;lt;TaskState | null&amp;gt;(null);&lt;br /&gt;
 const TaskDispatchContext = createContext&amp;lt;React.Dispatch&amp;lt;TaskAction&amp;gt; | null&amp;gt;(null);&lt;br /&gt;
 &lt;br /&gt;
 // Reducer&lt;br /&gt;
 function taskReducer(state: TaskState, action: TaskAction): TaskState {&lt;br /&gt;
    switch (action.type) {&lt;br /&gt;
       case &amp;#039;ADD&amp;#039;:&lt;br /&gt;
          return {&lt;br /&gt;
             ...state,&lt;br /&gt;
             tasks: [...state.tasks, { id: state.nextId, text: action.payload, completed: false }],&lt;br /&gt;
             nextId: state.nextId + 1,&lt;br /&gt;
          };&lt;br /&gt;
       case &amp;#039;REMOVE&amp;#039;:&lt;br /&gt;
          return { ...state, tasks: state.tasks.filter((t) =&amp;gt; t.id !== action.payload) };&lt;br /&gt;
       case &amp;#039;TOGGLE&amp;#039;:&lt;br /&gt;
          return {&lt;br /&gt;
             ...state,&lt;br /&gt;
             tasks: state.tasks.map((t) =&amp;gt;&lt;br /&gt;
                t.id === action.payload ? { ...t, completed: !t.completed } : t&lt;br /&gt;
             ),&lt;br /&gt;
          };&lt;br /&gt;
       default:&lt;br /&gt;
          return state;&lt;br /&gt;
    }&lt;br /&gt;
 }&lt;br /&gt;
 &lt;br /&gt;
 // Providerコンポーネント&lt;br /&gt;
 function TaskProvider({ children }: { children: ReactNode }) {&lt;br /&gt;
    const [state, dispatch] = useReducer(taskReducer, { tasks: [], nextId: 1 });&lt;br /&gt;
 &lt;br /&gt;
    return (&lt;br /&gt;
       &amp;lt;TaskStateContext.Provider value={state}&amp;gt;&lt;br /&gt;
          &amp;lt;TaskDispatchContext.Provider value={dispatch}&amp;gt;&lt;br /&gt;
             {children}&lt;br /&gt;
          &amp;lt;/TaskDispatchContext.Provider&amp;gt;&lt;br /&gt;
       &amp;lt;/TaskStateContext.Provider&amp;gt;&lt;br /&gt;
    );&lt;br /&gt;
 }&lt;br /&gt;
 &lt;br /&gt;
 // カスタムHook : StateとDispatchを個別に取得&lt;br /&gt;
 function useTaskState(): TaskState {&lt;br /&gt;
    const context = useContext(TaskStateContext);&lt;br /&gt;
    if (!context) throw new Error(&amp;#039;useTaskState must be used within TaskProvider&amp;#039;);&lt;br /&gt;
    return context;&lt;br /&gt;
 }&lt;br /&gt;
 &lt;br /&gt;
 function useTaskDispatch(): React.Dispatch&amp;lt;TaskAction&amp;gt; {&lt;br /&gt;
    const context = useContext(TaskDispatchContext);&lt;br /&gt;
    if (!context) throw new Error(&amp;#039;useTaskDispatch must be used within TaskProvider&amp;#039;);&lt;br /&gt;
    return context;&lt;br /&gt;
 }&lt;br /&gt;
 &lt;br /&gt;
 // 使用例&lt;br /&gt;
 function TaskList() {&lt;br /&gt;
    // 状態のみを購読するため、dispatchが変化しても再レンダリングされない&lt;br /&gt;
    const { tasks } = useTaskState();&lt;br /&gt;
    const dispatch = useTaskDispatch();&lt;br /&gt;
 &lt;br /&gt;
    return (&lt;br /&gt;
       &amp;lt;ul&amp;gt;&lt;br /&gt;
          {tasks.map((task) =&amp;gt; (&lt;br /&gt;
             &amp;lt;li key={task.id}&amp;gt;&lt;br /&gt;
                &amp;lt;span onClick={() =&amp;gt; dispatch({ type: &amp;#039;TOGGLE&amp;#039;, payload: task.id })}&amp;gt;&lt;br /&gt;
                   {task.text}&lt;br /&gt;
                &amp;lt;/span&amp;gt;&lt;br /&gt;
                &amp;lt;button onClick={() =&amp;gt; dispatch({ type: &amp;#039;REMOVE&amp;#039;, payload: task.id })}&amp;gt;削除&amp;lt;/button&amp;gt;&lt;br /&gt;
             &amp;lt;/li&amp;gt;&lt;br /&gt;
          ))}&lt;br /&gt;
       &amp;lt;/ul&amp;gt;&lt;br /&gt;
    );&lt;br /&gt;
 }&lt;br /&gt;
 &lt;br /&gt;
 function App() {&lt;br /&gt;
    return (&lt;br /&gt;
       &amp;lt;TaskProvider&amp;gt;&lt;br /&gt;
          &amp;lt;TaskList /&amp;gt;&lt;br /&gt;
       &amp;lt;/TaskProvider&amp;gt;&lt;br /&gt;
    );&lt;br /&gt;
 }&lt;br /&gt;
 &amp;lt;/syntaxhighlight&amp;gt;&lt;br /&gt;
&amp;lt;br&amp;gt;&amp;lt;br&amp;gt;&lt;br /&gt;
&lt;br /&gt;
== Immerライブラリとの組み合わせ ==&lt;br /&gt;
Immerライブラリを使用すると、ネストされた状態の更新をスプレッド演算子なしで簡単に記述できる。&amp;lt;br&amp;gt;&lt;br /&gt;
&amp;lt;br&amp;gt;&lt;br /&gt;
Immerの &amp;lt;code&amp;gt;produce&amp;lt;/code&amp;gt; 関数内では、状態を直接変更するような記述ができるが、実際には内部でイミュータブルな更新が行われる。&amp;lt;br&amp;gt;&lt;br /&gt;
&amp;lt;br&amp;gt;&lt;br /&gt;
&amp;lt;u&amp;gt;Immerを使用する場合は、&amp;lt;code&amp;gt;npm install immer&amp;lt;/code&amp;gt; コマンドを実行して、Immerパッケージをインストールする必要がある。&amp;lt;/u&amp;gt;&amp;lt;br&amp;gt;&lt;br /&gt;
&amp;lt;br&amp;gt;&lt;br /&gt;
 &amp;lt;syntaxhighlight lang=&amp;quot;typescript&amp;quot;&amp;gt;&lt;br /&gt;
 import { useReducer } from &amp;#039;react&amp;#039;;&lt;br /&gt;
 import { produce } from &amp;#039;immer&amp;#039;;&lt;br /&gt;
 &lt;br /&gt;
 interface User {&lt;br /&gt;
    id: number;&lt;br /&gt;
    name: string;&lt;br /&gt;
    email: string;&lt;br /&gt;
    address: {&lt;br /&gt;
       city: string;&lt;br /&gt;
       zipCode: string;&lt;br /&gt;
    };&lt;br /&gt;
 }&lt;br /&gt;
 &lt;br /&gt;
 interface UserState {&lt;br /&gt;
    users: Record&amp;lt;number, User&amp;gt;;&lt;br /&gt;
    selectedId: number | null;&lt;br /&gt;
 }&lt;br /&gt;
 &lt;br /&gt;
 type UserAction =&lt;br /&gt;
    | { type: &amp;#039;UPDATE_EMAIL&amp;#039;; id: number; payload: string }&lt;br /&gt;
    | { type: &amp;#039;UPDATE_CITY&amp;#039;; id: number; payload: string }&lt;br /&gt;
    | { type: &amp;#039;SELECT&amp;#039;; payload: number };&lt;br /&gt;
 &lt;br /&gt;
 // スプレッド演算子を使用した従来の記述&lt;br /&gt;
 function reducerWithSpread(state: UserState, action: UserAction): UserState {&lt;br /&gt;
    switch (action.type) {&lt;br /&gt;
       case &amp;#039;UPDATE_EMAIL&amp;#039;:&lt;br /&gt;
          return {&lt;br /&gt;
             ...state,&lt;br /&gt;
             users: {&lt;br /&gt;
                ...state.users,&lt;br /&gt;
                [action.id]: {&lt;br /&gt;
                   ...state.users[action.id],&lt;br /&gt;
                   email: action.payload,&lt;br /&gt;
                },&lt;br /&gt;
             },&lt;br /&gt;
          };&lt;br /&gt;
       case &amp;#039;UPDATE_CITY&amp;#039;:&lt;br /&gt;
          return {&lt;br /&gt;
             ...state,&lt;br /&gt;
             users: {&lt;br /&gt;
                ...state.users,&lt;br /&gt;
                [action.id]: {&lt;br /&gt;
                   ...state.users[action.id],&lt;br /&gt;
                   address: {&lt;br /&gt;
                      ...state.users[action.id].address,&lt;br /&gt;
                      city: action.payload,&lt;br /&gt;
                   },&lt;br /&gt;
                },&lt;br /&gt;
             },&lt;br /&gt;
          };&lt;br /&gt;
       default:&lt;br /&gt;
          return state;&lt;br /&gt;
    }&lt;br /&gt;
 }&lt;br /&gt;
 &lt;br /&gt;
 // Immerを使用した簡潔な記述&lt;br /&gt;
 const reducerWithImmer = produce((draft: UserState, action: UserAction) =&amp;gt; {&lt;br /&gt;
    switch (action.type) {&lt;br /&gt;
       case &amp;#039;UPDATE_EMAIL&amp;#039;:&lt;br /&gt;
          // draft を直接変更するように記述できる&lt;br /&gt;
          draft.users[action.id].email = action.payload;&lt;br /&gt;
          break;&lt;br /&gt;
       case &amp;#039;UPDATE_CITY&amp;#039;:&lt;br /&gt;
          // ネストが深い場合も簡潔に記述できる&lt;br /&gt;
          draft.users[action.id].address.city = action.payload;&lt;br /&gt;
          break;&lt;br /&gt;
       case &amp;#039;SELECT&amp;#039;:&lt;br /&gt;
          draft.selectedId = action.payload;&lt;br /&gt;
          break;&lt;br /&gt;
    }&lt;br /&gt;
 });&lt;br /&gt;
 &lt;br /&gt;
 function UserManager() {&lt;br /&gt;
    const [state, dispatch] = useReducer(reducerWithImmer, {&lt;br /&gt;
       users: {},&lt;br /&gt;
       selectedId: null,&lt;br /&gt;
    });&lt;br /&gt;
 &lt;br /&gt;
    return &amp;lt;div&amp;gt;...&amp;lt;/div&amp;gt;;&lt;br /&gt;
 }&lt;br /&gt;
 &amp;lt;/syntaxhighlight&amp;gt;&lt;br /&gt;
&amp;lt;br&amp;gt;&amp;lt;br&amp;gt;&lt;br /&gt;
&lt;br /&gt;
== よくある間違い ==&lt;br /&gt;
==== 状態の直接変更 ====&lt;br /&gt;
&amp;lt;code&amp;gt;reducer&amp;lt;/code&amp;gt; 内で既存の状態オブジェクトを直接変更してそのまま返すと、Reactが変更を検知できず、再レンダリングが発生しない。&amp;lt;br&amp;gt;&lt;br /&gt;
&amp;lt;u&amp;gt;常に新しいオブジェクトを返す必要がある。&amp;lt;/u&amp;gt;&amp;lt;br&amp;gt;&lt;br /&gt;
&amp;lt;br&amp;gt;&lt;br /&gt;
 &amp;lt;syntaxhighlight lang=&amp;quot;typescript&amp;quot;&amp;gt;&lt;br /&gt;
 // 正しい : 新しいオブジェクトを返す&lt;br /&gt;
 function goodReducer(state: State, action: Action): State {&lt;br /&gt;
    switch (action.type) {&lt;br /&gt;
       case &amp;#039;INCREMENT&amp;#039;:&lt;br /&gt;
          return { ...state, count: state.count + 1 }; // スプレッドで新しいオブジェクトを作成&lt;br /&gt;
       case &amp;#039;ADD_ITEM&amp;#039;:&lt;br /&gt;
          return { ...state, items: [...state.items, action.payload] }; // 新しい配列を作成&lt;br /&gt;
    }&lt;br /&gt;
 }&lt;br /&gt;
 &lt;br /&gt;
 // 誤り : 既存のオブジェクトを直接変更して返している&lt;br /&gt;
 function badReducer(state: State, action: Action): State {&lt;br /&gt;
    switch (action.type) {&lt;br /&gt;
       case &amp;#039;INCREMENT&amp;#039;:&lt;br /&gt;
          state.count += 1; // 直接変更&lt;br /&gt;
          return state;     // 同じオブジェクト参照を返す (変更を検知できない)&lt;br /&gt;
       case &amp;#039;ADD_ITEM&amp;#039;:&lt;br /&gt;
          state.items.push(action.payload); // 配列の直接変更&lt;br /&gt;
          return state;&lt;br /&gt;
    }&lt;br /&gt;
 }&lt;br /&gt;
 &amp;lt;/syntaxhighlight&amp;gt;&lt;br /&gt;
&amp;lt;br&amp;gt;&lt;br /&gt;
==== アクション型の不整合 ====&lt;br /&gt;
&amp;lt;code&amp;gt;dispatch&amp;lt;/code&amp;gt; に渡すアクションの &amp;lt;code&amp;gt;type&amp;lt;/code&amp;gt; 文字列と、&amp;lt;code&amp;gt;reducer&amp;lt;/code&amp;gt; の &amp;lt;code&amp;gt;case&amp;lt;/code&amp;gt; 文字列が一致しない場合、アクションが処理されない。&amp;lt;br&amp;gt;&lt;br /&gt;
&amp;lt;u&amp;gt;TypeScriptのDiscriminated Union型を使用することにより、この種のミスをコンパイル時に検知できる。&amp;lt;/u&amp;gt;&amp;lt;br&amp;gt;&lt;br /&gt;
&amp;lt;br&amp;gt;&lt;br /&gt;
 &amp;lt;syntaxhighlight lang=&amp;quot;typescript&amp;quot;&amp;gt;&lt;br /&gt;
 // 正しい : TypeScriptの型定義で整合性を保証する&lt;br /&gt;
 type Action = { type: &amp;#039;INCREMENT&amp;#039; } | { type: &amp;#039;DECREMENT&amp;#039; };&lt;br /&gt;
 &lt;br /&gt;
 dispatch({ type: &amp;#039;INCREMENT&amp;#039; });   // 型チェックにより一致が保証される&lt;br /&gt;
 &lt;br /&gt;
 // 誤り : caseの文字列が大文字なのに、dispatchは小文字&lt;br /&gt;
 dispatch(&amp;#039;increment&amp;#039;);             // string型で渡している&lt;br /&gt;
 dispatch({ type: &amp;#039;increment&amp;#039; });   // 小文字&lt;br /&gt;
 &lt;br /&gt;
 // reducer側&lt;br /&gt;
 case &amp;#039;INCREMENT&amp;#039;:                  // 大文字 (一致しない)&lt;br /&gt;
 &amp;lt;/syntaxhighlight&amp;gt;&lt;br /&gt;
&amp;lt;br&amp;gt;&lt;br /&gt;
==== Reducer内の副作用 ====&lt;br /&gt;
&amp;lt;code&amp;gt;reducer&amp;lt;/code&amp;gt; 関数は純粋関数でなければならない。&amp;lt;br&amp;gt;&lt;br /&gt;
API呼び出し、ローカルストレージへの書き込み、乱数生成、タイマの設定等の副作用を &amp;lt;code&amp;gt;reducer&amp;lt;/code&amp;gt; 内に記述してはいけない。&amp;lt;br&amp;gt;&lt;br /&gt;
&amp;lt;br&amp;gt;&lt;br /&gt;
&amp;lt;u&amp;gt;副作用は &amp;lt;code&amp;gt;useEffect&amp;lt;/code&amp;gt; 内で処理する。&amp;lt;/u&amp;gt;&amp;lt;br&amp;gt;&lt;br /&gt;
&amp;lt;br&amp;gt;&lt;br /&gt;
 &amp;lt;syntaxhighlight lang=&amp;quot;typescript&amp;quot;&amp;gt;&lt;br /&gt;
 // 誤り : reducer内で副作用を実行している&lt;br /&gt;
 function badReducer(state: State, action: Action): State {&lt;br /&gt;
    switch (action.type) {&lt;br /&gt;
       case &amp;#039;SAVE&amp;#039;:&lt;br /&gt;
          localStorage.setItem(&amp;#039;data&amp;#039;, JSON.stringify(state)); // 副作用&lt;br /&gt;
          fetch(&amp;#039;/api/save&amp;#039;, { method: &amp;#039;POST&amp;#039;, body: JSON.stringify(state) }); // 副作用&lt;br /&gt;
          return state;&lt;br /&gt;
       case &amp;#039;ADD&amp;#039;:&lt;br /&gt;
          return { ...state, id: Math.random() }; // 乱数生成も副作用&lt;br /&gt;
    }&lt;br /&gt;
 }&lt;br /&gt;
 &lt;br /&gt;
 // 正しい : 副作用はuseEffectで処理する&lt;br /&gt;
 function goodReducer(state: State, action: Action): State {&lt;br /&gt;
    switch (action.type) {&lt;br /&gt;
       case &amp;#039;ADD&amp;#039;:&lt;br /&gt;
          return { ...state, items: [...state.items, action.payload] };&lt;br /&gt;
       default:&lt;br /&gt;
          return state;&lt;br /&gt;
    }&lt;br /&gt;
 }&lt;br /&gt;
 &lt;br /&gt;
 function MyComponent() {&lt;br /&gt;
    const [state, dispatch] = useReducer(goodReducer, initialState);&lt;br /&gt;
 &lt;br /&gt;
    // 副作用はuseEffectで処理する&lt;br /&gt;
    useEffect(() =&amp;gt; {&lt;br /&gt;
       localStorage.setItem(&amp;#039;data&amp;#039;, JSON.stringify(state));&lt;br /&gt;
    }, [state]);&lt;br /&gt;
 &lt;br /&gt;
    return &amp;lt;div&amp;gt;...&amp;lt;/div&amp;gt;;&lt;br /&gt;
 }&lt;br /&gt;
 &amp;lt;/syntaxhighlight&amp;gt;&lt;br /&gt;
&amp;lt;br&amp;gt;&amp;lt;br&amp;gt;&lt;br /&gt;
&lt;br /&gt;
== 関連情報 ==&lt;br /&gt;
* [[Reactの基礎 - Hooksの基礎]]&lt;br /&gt;
* [[Reactの基礎 - useState]]&lt;br /&gt;
* [[Reactの基礎 - useContext]]&lt;br /&gt;
* [[Reactの基礎 - カスタムHook]]&lt;br /&gt;
&amp;lt;br&amp;gt;&amp;lt;br&amp;gt;&lt;br /&gt;
&lt;br /&gt;
&lt;br /&gt;
{{#seo:&lt;br /&gt;
|title={{PAGENAME}} : Exploring Electronics and SUSE Linux | MochiuWiki&lt;br /&gt;
|keywords=MochiuWiki,Mochiu,Wiki,Mochiu Wiki,Electric Circuit,Electric,pcb,Mathematics,AVR,TI,STMicro,AVR,ATmega,MSP430,STM,Arduino,Xilinx,FPGA,Verilog,HDL,PinePhone,Pine Phone,Raspberry,Raspberry Pi,C,C++,C#,Qt,Qml,MFC,Shell,Bash,Zsh,Fish,SUSE,SLE,Suse Enterprise,Suse Linux,openSUSE,open SUSE,Leap,Linux,uCLnux,電気回路,電子回路,基板,プリント基板,React,JavaScript,TypeScript,TSX,useReducer,useState,useContext,Hooks,Reducer,Redux,状態管理,Immer,Discriminated Union,dispatch,グローバル状態&lt;br /&gt;
|description={{PAGENAME}} - 電子回路とSUSE Linuxに関する情報 | This page is {{PAGENAME}} in our wiki about electronic circuits and SUSE Linux&lt;br /&gt;
|image=/resources/assets/MochiuLogo_Single_Blue.png&lt;br /&gt;
}}&lt;br /&gt;
&lt;br /&gt;
__FORCETOC__&lt;br /&gt;
[[カテゴリ:Rust]][[カテゴリ:Web]]&lt;/div&gt;</summary>
		<author><name>Wiki</name></author>
	</entry>
</feed>