摘要:函數的作用是,返回一個最終的函數,做的事情是得到一個由多個不同函數作為可以自定義的對象。把數據放到一個對象列表中,每個數據用作為主鍵。不同類型的對象通過引用數據,這樣數據發生改變的時候,只需要修改一處地方,減少數據冗余或者混用。
Redux初見
本文記錄的是自己對redux的學習和理解,希望可以簡潔易懂,入門redux,一步步的走進redux!
Redux是什么核心概念Redux是JavaScript應用的狀態容器,提供可預測化的狀態管理,讓你構建一致化的應用,運行于不同的環境(客戶端、服務器、原生應用),并且易于測試。不僅于此,它還提供超爽的開發體驗。
【store】保存應用狀態全局單一
【state】所有state以key-value的形式存儲在store中
【action】描述發生什么的對象,觸發action是唯一改變state的方式,action本質上是一個普通js對象,必必須一個type字段,值為字符串常量
【action creator】創建action
【reducer】描述action如何改變state
Redux常用方法 createStore(reducer, [initialState])創建一個Redux store來以存放應用中所有的state。
combineReducers(reducers)隨著應用復雜度上升,數據源逐漸變混亂,導致組件內部數據調用十分復雜,會產生數據冗余或者混用等情況,需要對reducer函數進行拆分,拆分后的每一模塊獨立負責管理state的一部分。combineReducers函數的作用是,返回一個最終的rootReducer函數,rootReducer做的事情是得到一個由多個不同reducer函數作為value(key可以自定義)的對象。
const module1Reducer = combineReducers( module1_key1: module1_reducer1 ); const rootReducer = combineReducers( key1: reducer1, module1: module1Reducer ); const store = createStore(rootReducer); // store中保存的state結構如下 { key1: reducer1(state.key1, action), module1: { module1_key1: (state.module1.module1_key1, action) } }
總結一下,state對象的結構由傳入的多個reducer的key決定,可以根據模塊拆分的細粒度,考慮是否需要嵌套使用combineReducers,整個應用的數據大致分來兩類:普通data和ui狀態
+ data - 服務器響應的數據 - 緩存數據 - 本地尚未持久化到服務器的數據 - 用戶輸入 - ... + ui狀態 - 激活的路由 - 被選中的Tab標簽 - 是否顯示加載動畫 - 分頁器狀態 - ...
設計state結構時,盡量把state范式化,不要存在嵌套不同類型的對象的情況。把數據放到一個對象(列表)中,每個數據用id作為主鍵。不同類型的對象通過id引用數據,這樣數據發生改變的時候,只需要修改一處地方,減少數據冗余或者混用。
applyMiddleware(...middlewares)首先要介紹一下什么是middlewares,middlewares用于包裝store.dispatch,擴展其功能,在發起action之后,到達reducer之前執行一些邏輯,有點類似是aop的一種實現。
applyMiddleware大致實現:
暫存redux store提供的dispatch
把dispatch作為實參,傳給middleware執行之后返回的函數A
執行函數A,返回包裝過的dispatch,覆蓋原來的store.dispatch
function applyMiddleware(store, middlewares) { middlewares = middlewares.slice() middlewares.reverse() // 暫存dispatch let dispatch = store.dispatch // 包裝dispatch middlewares.forEach(middleware => dispatch = middleware(store)(dispatch) ) return {...store, { dispatch }) }
理解了applyMiddleware的邏輯,自定義一個middleware大致如下
function(store){ // pass store.dispatch to next return function(next){ // return dispatch return function(action){ // implement middleware logic } } }bindActionCreators(actionCreators, dispatch)
參數actionCreators如果為函數 把 action creators 轉成擁有同名keys的對象,但使用 dispatch 把每個action creator包圍起來,返回新的對象
參數actionCreators如果為對象,若actionCreators[key]為函數,用dispatch把每個 actionCreators[key]包圍起來,返回新的對象
compose(...functions)組合store enhance,applyMiddleware 和 redux-devtools就是store enhance
React-redux橋接Redux本身只提供應用狀態和數據流管理,除了和React一起用外,還支持其它界面庫,且沒有任何依賴。要在React的項目中使用Redux,比較好的方式是借助react-redux這個庫來做連接.
provider為整個應用提供store數據,做的事情是把store作為props傳遞到每一個被connet()包裝的組件
connect([mapStateToProps], [mapDispatchToProps], [mergeProps], [options])connect(...args)(component)返回一個與Redux store連接的組件類,下面簡單講解一下connect方法的參數
state是store中整個應用的state
ownProps高階函數包裝過的組件的props
該回調函數必須返回一個純對象,這個對象會與被包裝的組件的props做merge合并
mapStateToProps可以做一些數據的format, filter,compose操作,保證數據在組件層面的方便使用
mapDispatchToProps參數為Object, 每個定義在該對象的函數都將被當作Redux action creator,其中所定義的方法名將作為屬性名,合并到被包裝的組件的props中,實現的效果:執行component.prop.checkout()實際上是dispatch了一個action,這樣做的好處是component與redux的解耦,component根本不知道redux的存在。
// action creator返回對象 mapDispatchToProps = { // action creator checkout: function actionCreator(productId){ return { type: types.ADD_TO_CART, productId } } } // action creator返回函數,thunk mapDispatchToProps = { // action creator checkout: function actionCreator(productId){ action creator 返回一個thunk, thunk的參數為 dispatch 和 getState return (dispatch, getState) => {? if (getState().products.byId[productId].inventory > 0) {? dispatch(addToCartUnsafe(productId))? }? } } } // 最終被綁定的組件props component.props.checkout = function () { return dispatch(actionCreator.apply(undefined, arguments)); }
mapDispatchToProps參數為Function
mapDispatchToProps(dispatch, [ownProps]) = function(){ return { checkout: () => {? dispatch(actionCreator())? } } } // 最終被綁定的組件.props component.props.checkout = function () { return dispatch(actionCreator.apply(undefined, arguments)); } // 使用bindActionCreators mapDispatchToProps(dispatch, [ownProps]) = function(){ return bindActionCreators(actionCreator, dispatch) } // 最終被綁定的組件props component.props.actionCreator = function () { return dispatch(actionCreator.apply(undefined, arguments)); } // 使用 bindActionCreators mapDispatchToProps(dispatch, [ownProps]) = function(){ return bindActionCreators({ ‘checkout’: actionCreator }, dispatch) } // 最終被綁定的組件props component.props.checkout = function () { return dispatch(actionCreator.apply(undefined, arguments)); }使用redux之后應用的數據流向
dispatch(actionCreator)?=> Reducer => (state, action) => state
用戶操作或者網絡請求 store.dispatch(action)
redux store調用傳入的rootReducer
redux 執行全部的reducer,把每個reducer執行得到的結果輸出合成一個新的對象
store 存儲rootReducer返回的值,更新currentState
文章版權歸作者所有,未經允許請勿轉載,若此文章存在違規行為,您可以聯系管理員刪除。
轉載請注明本文地址:http://m.specialneedsforspecialkids.com/yun/80116.html
摘要:系列教程,持續更新系列教程應用初見系列教程處理靜態文件系列教程使用模板引擎系列教程路由控制中間件系列教程綜合搭建登錄注冊頁面系列教程實現登錄注冊功能這篇教程主要介紹構建服務器,簡單引用本教程的版本要格外注意版本號案例簡單利用搭建服務器文件夾 koa2系列教程,持續更新 koa2系列教程:koa2應用初見 koa2系列教程:koa2處理靜態文件 koa2系列教程:koa2使用模板引擎 ...
摘要:在學習的過程中時常會聽到一個名次函數式編程,那么究竟什么是函數式編程,函數式編程又有什么優點,這就在這篇博客進行一個簡單的總結吧主要內容函數式編程的概念函數式編程的優點與示例什么是函數式編程首先,我們放下編程的概念,我們來看函數。 在學習 JS 的過程中時常會聽到一個名次——函數式編程,那么究竟什么是函數式編程,函數式編程又有什么優點,這就在這篇博客進行一個簡單的總結吧~ 主要內容: 函...
摘要:準備好數據節點節點節點坐標節點坐標邊節點,從哪里出發節點,到哪里結束初始化對象容器渲染位置,表示渲染到圖表的中間位置畫布高渲染數據這是渲染出來的效果。鏈接線會以元素為基準。繪制元素時,需要在初始化對象的時候,指定。 hello world // 1. 準備好數據 // node(節點) let nodes = [ { id: 1, // 節點 id ...
閱讀 1014·2021-11-22 13:52
閱讀 1448·2021-11-19 09:40
閱讀 3174·2021-11-16 11:44
閱讀 1276·2021-11-15 11:39
閱讀 3909·2021-10-08 10:04
閱讀 5367·2021-09-22 14:57
閱讀 3106·2021-09-10 10:50
閱讀 3188·2021-08-17 10:13