Modern React with Redux:搭配 Redux

Published on
modern-react-with-redux

React 主要處理顯示與使用者互動,不是專門用來維護數據,所以可依據專案、資料來源和使用者操作的複雜度,選擇是否搭配 Redux。

如果把 Redux 的流程想像成一間保險公司:

1. 用戶填寫表格,申請投保、理賠、解約(action creator)
2. 每一張表格都是一個動作(action)
3. 統一轉交給特定人(dispatch)
4. 表格會派送到各部門受理(reducers)
5. 根據表格內容做不同的處理,這些更新或修改也被統一更新到公司的數據中心(store)
  • Action Creator:用來創建 Action(一般來的 JavaScript 物件)的函數
  • Action:這個 JS 物件必須帶有 type 和可選的 payload 屬性,type 代表要對資料進行的操作、payload 則是該操作要帶上的參數
  • Dispatch:負責轉交 action 的人(和 Redux 溝通的關鍵
  • Reducer:負責處理資料的函數,可以想像 reducers 是一個個負責不同業務的部門,他們接收一個 action、查看目前的 state,然後根據 action 來更新資料
  • Store:公司的數據中心,每個部門(reducer)創建時的初始資料會被整合到 Store 裡面的 state object,而 reducers 處理更新完的資料也會同步到這裡,不用一一去各部門查看

Store 是集中保存數據的地方,整個 App 只會有一個 Store,Redux 提供 createStore 這個函數,用来生成 Store。

import { createStore } from 'redux'
const store = createStore(fn)

要和 Store 溝通,只能透過它提供的少數方法進行操作

  • 使用 getState 獲得資料
  • 使用 dispatch 將 action 派送給 Store
// 通過 getState() 拿到 Store 裡面的 state object
const state = store.getState()

// 使用 dispatch() 將指令送到 Store
const action = {
  type: 'ADD_TODO',
  payload: 'Learn Redux',
}
store.dispatch(action)

Store 在收到 action 後,會計算並返回一個新的 state,這個計算過程就是 reducer。

要使用 store.dispatch() 來觸發 Reducer 自動執行,在一開始生成 Store 時,我們就要將 reducers 作為參數傳入 createStore 方法,這邊可以想像成是註冊 => 讓 Store 知道有哪些 reducers 函數。

因為整個 Store 只會有一個 state object,裡面包含各式各樣的屬性,不同 Reducer 函數會負責生成不同的 state,為了避免龐雜的程式碼,Redux 有提供一個 combineReducers 方法用來拆分 Reducer。

只要定義各個子 Reducer 函數,然後用這個方法將他們合成成一個大的 Reducer 就好,根據 State 的 key 去執行對應的子 Reducer,再將結果合併成一個大的 state object 更新。

import { createStore, combineReducers } from 'redux'

const ourDepartments = combineReducers({
  accouting: accouting, // 會記部門
  claimsHistory: claimsHistory, // 理賠部門
  policies: policies, // 行政部門
})

const store = createStore(ourDepartments)
// 可以將不同的 state 整合一起
// 透過不同的 key name 形成不同的 property,儲存在 store

使用 Redux 的好處是,只會有少量方法可以修改到數據,雖然一開始設置很複雜,不過隨著專案變大、數據來源多樣,複雜程度不會像純用 React 到後期會面臨難以管理的問題。


在 React 中使用 Redux

安裝 Redux 和 React-Redux,裡面有很多輔助函數/元件,協助結合兩者

npm install --save redux react-redux

參考 Redux 入门教程(三),React-Redux 將元件分成兩類別

  • UI 元件(presentational component)
  • 容器元件(container component)

容器元件,負責管理資料和邏輯。

如果一個元件除了要顯示 UI、同時又兼具邏輯部分,會被拆分成這樣的架構:「外層是容器元件,裡面包了一個 UI 元件,前者負責和外部的溝通,將資料傳給後者來顯示

從這層觀點切入,來看 Provider 和 Connect 元件的話,就不那麼難理解他們的用法了:

  1. 首先在 Store 裡面已存放所有的 state 和 reducers,我們從 react-redux 引入新元件 Provider,將 Store 引用傳給它
// index.js
import React from 'react'
import ReactDOM from 'react-dom'
import { Provider } from 'react-redux'
import { createStore } from 'redux'
import App from './components/App'
import reducers from './reducers'

ReactDOM.render(
  <Provider store={createStore(reducers)}>
    <App />
  </Provider>,
  document.getElementById('root')
)

這樣一來,所有在 Provider 下面的元件會默認可以從其拿到 Store 裡面的 state,只要讓 Provider 包在根元件外面!原理是利用 React 的 context system 🔸,允許父層和底下的子元件直接聯繫、就算中間隔了好幾個元件。

  1. 接著在子元件設置上,使用 connect 函數來讓 UI 元件生成為容器元件
import { connect } from 'react-redux'
export default connect()(TodoList) // TodoList 自定義的 UI 元件

connect 函數,會需要接受參數來定義邏輯部分、設置與 Store 的溝通

  • 輸入邏輯:外部資料(state object)如何轉換為 UI 元件的 props
  • 輸出邏輯:使用者的操作如何轉換成 action,從 UI 元件發送給 Store
import { connect } from 'react-redux'

const default connect(
  mapStateToProps,
  mapDispatchToProps
)(TodoList)

第一個參數 mapStateToProps

定義 UI 元件如何取得 state object 裡面的資料,映射轉換為 props。如果這邊省略 mapStateToProps 參數的話,UI 元件就不会訂閱 Store,換句話說就是 state 更新不會引起 UI 上的變化。

const mapStateToProps = (state) => {
  // 參數拿到 state
  return { songs: state.songsReducer }
}

export default connect(mapStateToProps)(SongList)
// 定義 SongList 元件要從 store 拿的資料,可以不用是全部的 state
// 在 SongList 元件的 props 上會多一個自定義的屬性 "songs"

第二個參數 mapDispatchToProps

定義使用者的哪些操作會作為 Action 發送給 Store,將他們映射為 props 來給 UI 元件使用,可以是 function、也可以是 object

// function 的話,會得到 dispatch 參數
const mapDispatchToProps = (dispatch) => {
  return {
    addFeatures: (feature) => dispatch(addFeatures(feature)),
    removeFeatures: (feature) => dispatch(removeFeatures(feature)),
  }
}

物件的話,其實就是放 Action Creator

// 原本 selectSong 只是一個普通的 action creator,沒有和 Redux 串在一起
import { selectSong } from '../actions'

export default connect(mapStateToProps, {
  selectSong: selectSong,
})(SongList)

直接呼叫 Action Creator 是沒有作用的,只會做為普通的函數執行,Redux 不會沒事通靈、知道你正在調用 Action Creator,所以才要額外透過 connect 連結起來

想更新 State,就得調用 dispatch function 發送到 Store,而這邊透過 connect 連結起來,它會查看第二個參數裡面帶的物件,將物件裡面的函數包裝成另一個 JS function,等於會先自動調用 dispatch,執行發送到 Store。

最後補充

所謂 connect 語法,說白了是返回一個函數,我們再接著呼叫這個返回函數,傳入自定義的 UI 元件

function connect() {
  return function () {
    return 'hi there!'
  }
}

connect()() // 'hi there!'

使用 connect 方法後,實際上還是得到元件,不過會擁有一些額外的邏輯設置、可以和 Store 進行溝通。一旦 state 有所改變就會自動通知 connect,訂閱它的 UI 元件就會收到更新、將東西傳到我們自定義的元件。


本篇旨在整理 Modern React with Redux 第十七章的內容,非完整的教學文,建議搭配其他資料學習