applyMiddleware(...middlewares)
使用包含自定義功能的 middleware 來擴展 Redux 是一種推薦的方式。Middleware 可以讓你包裝 store 的dispatch
?方法來達到你想要的目的。同時, middleware 還擁有“可組合”這一關鍵特性。多個 middleware 可以被組合到一起使用,形成 middleware 鏈。其中,每個 middleware 都不需要關心鏈中它前后的 middleware 的任何信息。
Middleware 最常見的使用場景是無需引用大量代碼或依賴類似?Rx?的第三方庫實現異步 actions。這種方式可以讓你像 dispatch 一般的 actions 那樣 dispatch?異步 actions。
例如,redux-thunk?支持 dispatch function,以此讓 action creator 控制反轉。被 dispatch 的 function 會接收?dispatch
?作為參數,并且可以異步調用它。這類的 function 就稱為?thunk。另一個 middleware 的示例是?redux-promise。它支持 dispatch 一個異步的?Promise?action,并且在 Promise resolve 后可以 dispatch 一個普通的 action。
Middleware 并不需要和?createStore
?綁在一起使用,也不是 Redux 架構的基礎組成部分,但它帶來的益處讓我們認為有必要在 Redux 核心中包含對它的支持。因此,雖然不同的 middleware 可能在易用性和用法上有所不同,它仍被作為擴展?dispatch
?的唯一標準的方式。
參數
...middlewares
?(arguments): 遵循 Redux?middleware API?的函數。每個 middleware 接受?Store
?的dispatch
?和?getState
?函數作為命名參數,并返回一個函數。該函數會被傳入 被稱為?next
?的下一個 middleware 的 dispatch 方法,并返回一個接收 action 的新函數,這個函數可以直接調用next(action)
,或者在其他需要的時刻調用,甚至根本不去調用它。調用鏈中最后一個 middleware 會接受真實的 store 的?dispatch
?方法作為?next
?參數,并借此結束調用鏈。所以,middleware 的函數簽名是?({?getState,?dispatch?})?=>?next?=>?action
。
返回值
(Function) 一個應用了 middleware 后的 store enhancer。這個 store enhancer 就是一個函數,并且需要應用到?createStore
。它會返回一個應用了 middleware 的新的?createStore
。
示例: 自定義 Logger Middleware


1 import { createStore, applyMiddleware } from 'redux' 2 import todos from './reducers' 3 4 function logger({ getState }) { 5 return (next) => (action) => { 6 console.log('will dispatch', action) 7 8 // 調用 middleware 鏈中下一個 middleware 的 dispatch。 9 let returnValue = next(action) 10 11 console.log('state after dispatch', getState()) 12 13 // 一般會是 action 本身,除非 14 // 后面的 middleware 修改了它。 15 return returnValue 16 } 17 } 18 19 let createStoreWithMiddleware = applyMiddleware(logger)(createStore) 20 let store = createStoreWithMiddleware(todos, [ 'Use Redux' ]) 21 22 store.dispatch({ 23 type: 'ADD_TODO', 24 text: 'Understand the middleware' 25 }) 26 // (將打印如下信息:) 27 // will dispatch: { type: 'ADD_TODO', text: 'Understand the middleware' } 28 // state after dispatch: [ 'Use Redux', 'Understand the middleware' ]
示例: 使用 Thunk Middleware 來做異步 Action


1 import { createStore, combineReducers, applyMiddleware } from 'redux' 2 import thunk from 'redux-thunk' 3 import * as reducers from './reducers' 4 5 // 調用 applyMiddleware,使用 middleware 增強 createStore: 6 let createStoreWithMiddleware = applyMiddleware(thunk)(createStore) 7 8 // 像原生 createStore 一樣使用。 9 let reducer = combineReducers(reducers) 10 let store = createStoreWithMiddleware(reducer) 11 12 function fetchSecretSauce() { 13 return fetch('https://www.google.com/search?q=secret+sauce') 14 } 15 16 // 這些是你已熟悉的普通 action creator。 17 // 它們返回的 action 不需要任何 middleware 就能被 dispatch。 18 // 但是,他們只表達「事實」,并不表達「異步數據流」 19 20 function makeASandwich(forPerson, secretSauce) { 21 return { 22 type: 'MAKE_SANDWICH', 23 forPerson, 24 secretSauce 25 } 26 } 27 28 function apologize(fromPerson, toPerson, error) { 29 return { 30 type: 'APOLOGIZE', 31 fromPerson, 32 toPerson, 33 error 34 } 35 } 36 37 function withdrawMoney(amount) { 38 return { 39 type: 'WITHDRAW', 40 amount 41 } 42 } 43 44 // 即使不使用 middleware,你也可以 dispatch action: 45 store.dispatch(withdrawMoney(100)) 46 47 // 但是怎樣處理異步 action 呢, 48 // 比如 API 調用,或者是路由跳轉? 49 50 // 來看一下 thunk。 51 // Thunk 就是一個返回函數的函數。 52 // 下面就是一個 thunk。 53 54 function makeASandwichWithSecretSauce(forPerson) { 55 56 // 控制反轉! 57 // 返回一個接收 `dispatch` 的函數。 58 // Thunk middleware 知道如何把異步的 thunk action 轉為普通 action。 59 60 return function (dispatch) { 61 return fetchSecretSauce().then( 62 sauce => dispatch(makeASandwich(forPerson, sauce)), 63 error => dispatch(apologize('The Sandwich Shop', forPerson, error)) 64 ) 65 } 66 } 67 68 // Thunk middleware 可以讓我們像 dispatch 普通 action 69 // 一樣 dispatch 異步的 thunk action。 70 71 store.dispatch( 72 makeASandwichWithSecretSauce('Me') 73 ) 74 75 // 它甚至負責回傳 thunk 被 dispatch 后返回的值, 76 // 所以可以繼續串連 Promise,調用它的 .then() 方法。 77 78 store.dispatch( 79 makeASandwichWithSecretSauce('My wife') 80 ).then(() => { 81 console.log('Done!') 82 }) 83 84 // 實際上,可以寫一個 dispatch 其它 action creator 里 85 // 普通 action 和異步 action 的 action creator, 86 // 而且可以使用 Promise 來控制數據流。 87 88 function makeSandwichesForEverybody() { 89 return function (dispatch, getState) { 90 if (!getState().sandwiches.isShopOpen) { 91 92 // 返回 Promise 并不是必須的,但這是一個很好的約定, 93 // 為了讓調用者能夠在異步的 dispatch 結果上直接調用 .then() 方法。 94 95 return Promise.resolve() 96 } 97 98 // 可以 dispatch 普通 action 對象和其它 thunk, 99 // 這樣我們就可以在一個數據流中組合多個異步 action。 100 101 return dispatch( 102 makeASandwichWithSecretSauce('My Grandma') 103 ).then(() => 104 Promise.all([ 105 dispatch(makeASandwichWithSecretSauce('Me')), 106 dispatch(makeASandwichWithSecretSauce('My wife')) 107 ]) 108 ).then(() => 109 dispatch(makeASandwichWithSecretSauce('Our kids')) 110 ).then(() => 111 dispatch(getState().myMoney > 42 ? 112 withdrawMoney(42) : 113 apologize('Me', 'The Sandwich Shop') 114 ) 115 ) 116 } 117 } 118 119 // 這在服務端渲染時很有用,因為我可以等到數據 120 // 準備好后,同步的渲染應用。 121 122 import { renderToString } from 'react-dom/server' 123 124 store.dispatch( 125 makeSandwichesForEverybody() 126 ).then(() => 127 response.send(renderToString(<MyApp store={store} />)) 128 ) 129 130 // 也可以在任何導致組件的 props 變化的時刻 131 // dispatch 一個異步 thunk action。 132 133 import { connect } from 'react-redux' 134 import { Component } from 'react' 135 136 class SandwichShop extends Component { 137 componentDidMount() { 138 this.props.dispatch( 139 makeASandwichWithSecretSauce(this.props.forPerson) 140 ) 141 } 142 143 componentWillReceiveProps(nextProps) { 144 if (nextProps.forPerson !== this.props.forPerson) { 145 this.props.dispatch( 146 makeASandwichWithSecretSauce(nextProps.forPerson) 147 ) 148 } 149 } 150 151 render() { 152 return <p>{this.props.sandwiches.join('mustard')}</p> 153 } 154 } 155 156 export default connect( 157 state => ({ 158 sandwiches: state.sandwiches 159 }) 160 )(SandwichShop)
小貼士
-
Middleware 只是包裝了 store 的?
dispatch
?方法。技術上講,任何 middleware 能做的事情,都可能通過手動包裝?dispatch
?調用來實現,但是放在同一個地方統一管理會使整個項目的擴展變的容易得多。 -
如果除了?
applyMiddleware
,你還用了其它 store enhancer,一定要把?applyMiddleware
?放到組合鏈的前面,因為 middleware 可能會包含異步操作。比如,它應該在?redux-devtools?前面,否則 DevTools 就看不到 Promise middleware 里 dispatch 的 action 了。 -
如果你想有條件地使用 middleware,記住只 import 需要的部分:
?


1 let middleware = [ a, b ] 2 if (process.env.NODE_ENV !== 'production') { 3 let c = require('some-debug-middleware') 4 let d = require('another-debug-middleware') 5 middleware = [ ...middleware, c, d ] 6 } 7 const createStoreWithMiddleware = applyMiddleware(...middleware)(createStore)
-
這樣做有利于打包時去掉不需要的模塊,減小打包文件大小。
-
有想過?
applyMiddleware
?本質是什么嗎?它肯定是比 middleware 還強大的擴展機制。實際上,applyMiddleware
?只是被稱為 Redux 最強大的擴展機制的?store enhancers?中的一個范例而已。你不太可能需要實現自己的 store enhancer。另一個 store enhancer 示例是?redux-devtools。Middleware 并沒有 store enhancer 強大,但開發起來卻是更容易的。 -
Middleware 聽起來比實際難一些。真正理解 middleware 的唯一辦法是了解現有的 middleware 是如何工作的,并嘗試自己實現。需要的功能可能錯綜復雜,但是你會發現大部分 middleware 實際上很小,只有 10 行左右,是通過對它們的組合使用來達到最終的目的。
?