Redux04 React + Redux 最佳實踐
標簽: TypeScript
本文由 簡悅 SimpRead 轉碼, 原文地址 https://github.com/sorrycc/blog/issues/1
更新:我們基于此最佳實踐做了一個封裝方案:dva,可以簡化使用 redux 和 redux-saga 時很多繁雜的操作。
前端變化雖快,但其實一直都圍繞這幾個概念在轉:
- URL - 訪問什么頁面
- Data - 顯示什么信息
- View - 頁面長成什么樣
- Action - 對頁面做了什么操作
- API Server - Data 數據的來源
在 Redux 的生態圈內,每個環節有多種方案,比如 Data 可以是 immutable
或者 plain object
,在你選了 immutable
之后,用 immutable.js 還是 seamless-immutable,以及是否用 redux-immutable 來輔助數據修改,都需要選擇。
本文總結目前 react + redux 的最佳實踐,解釋原因,并提供可選方案。
心急的朋友可以直接看代碼:https://github.com/sorrycc/github-stars
一、URL > Data
需求
routing
選擇
react-router + react-router-redux: 前者是業界標準,后者可以同步 route 信息到 state,這樣你可以在 view 根據 route 信息調整展現,以及通過 action 來修改 route 。
可選
無
二、Data
需求
為 redux 提供數據源,修改容易。
方案
plain object
: 配合 combineReducer 已經可以滿足需求。
同時在組織 Store 的時候,層次不要太深,盡量保持在 2 - 3 層。如果層次深,可以考慮用 updeep 來輔助修改數據。
可選
immutable.js: 通過自定義的 api 來操作數據,需要額外的學習成本。不熟悉 immutable.js 的可以先嘗試用 seamless-immutable,JavaScript 原生接口,無學習門檻。
另外,不推薦用 redux-immutable 以及 redux-immutablejs,一是沒啥必要,具體看他們的實現就知道了,都比較簡單;更重要的是他們都改寫了 combineReducer
,會帶來潛在的一些兼容問題。
三、Data > View
需求
數據的過濾和篩選。
方案
reselect: store 的 select 方案,用于提取數據的篩選邏輯,讓 Component 保持簡單。選 reselct 看重的是 可組合特性
和 緩存機制
。
可選
無
四、View 之 CSS 方案
需求
合理的 CSS 方案,考慮團隊協作。
方案
css-modules: 配合 webpack 的 css-loader 進行打包,會為所有的 class name 和 animation name 加 local scope,避免潛在沖突。
直接看代碼:
Header.jsx
import style from './Header.less';
export default () => <div className={style.normal} />;
Header.less
.normal { color: red; }
編譯后,文件中的 style.normal
和 .normal
在會被重命名為類似 Header__normal___VI1de
。
可選
bem, rscss ,這兩個都是基于約定的方案。但基于約定會帶來額外的學習成本和不遍,比如 rscss 要求所有的 Component 都是兩個詞的連接,比如 Header
就必須換成類似 HeaderBox
這樣。
radium,inline css 方案,沒研究。
五、Action <> Store,業務邏輯處理
需求
統一處理業務邏輯,尤其是異步的處理。
方案
redux-saga: 用于管理 action,處理異步邏輯。可測試、可 mock、聲明式的指令。
可選
redux-loop: 適用于相對簡單點的場景,可以組合異步和同步的 action 。但他有個問題是改寫了 combineReducer
,會導致一些意想不到的兼容問題,比如我在特定場景下用不了 redux-devtool 。
redux-thunk, redux-promise 等: 相對原始的異步方案,適用于更簡單的場景。在 action 需要組合、取消等操作時,會不好處理。
saga 入門
在 saga 之前,你可能會在 action creator 里處理業務邏輯,雖然能跑通,但是難以測試。比如:
// action creator with thunking
function createRequest () {
return (dispatch, getState) => {
dispatch({ type: 'REQUEST_STUFF' });
someApiCall(function(response) {
// some processing
dispatch({ type: 'RECEIVE_STUFF' });
});
};
}
然后組件里可能這樣:
function onHandlePress () {
this.props.dispatch({ type: 'SHOW_WAITING_MODAL' });
this.props.dispatch(createRequest());
}
這樣通過 redux state 和 reducer 把所有的事情串聯到起來。
但問題是:
Code is everywhere.
通過 saga,你只需要觸發一個 action 。
function onHandlePress () {
// createRequest 觸發 action `BEGIN_REQUEST`
this.props.dispatch(createRequest());
}
然后所有后續的操作都通過 saga 來管理。
function *hello() {
// 等待 action `BEGIN_REQUEST`
yield take('BEGIN_REQUEST');
// dispatch action `SHOW_WAITING_MODAL`
yield put({ type: 'SHOW_WAITING_MODAL' });
// 發布異步請求
const response = yield call(myApiFunctionThatWrapsFetch);
// dispatch action `PRELOAD_IMAGES`, 附上 response 信息
yield put({ type: 'PRELOAD_IMAGES', response.images });
// dispatch action `HIDE_WAITING_MODAL`
yield put({ type: 'HIDE_WAITING_MODAL' });
}
可以看出,調整之后的代碼有幾個優點:
- 所有業務代碼都存于 saga 中,不再散落在各處
- 全同步執行,就算邏輯再復雜,看起來也不會亂
六、Data <-> API Server
需求
異步請求。
方案
isomorphic-fetch: 便于在同構應用中使用,另外同時要寫 node 和 web 的同學可以用一個庫,學一套 api 。
然后通過 async
+ await
組織代碼。
示例代碼:
import fetch from 'isomorphic-fetch';
export async function fetchUser(uid) {
return await fetch(`/users/${uid}`).then(res => res.json());
};
可選
最終
(完)
智能推薦
redux與react-redux
redux 概述 redux是一個javascript容器,用于進行全局的狀態管理 三大核心 單一數據源 整個應用的state被存儲在一個object tree中,并且這個object tree只存在于唯一一個store中 state是只讀的 唯一改變state的方法就是觸發action,action是一個用于描述已發生時間的普通對象 視圖和網絡請求不能直接去修改state,只表達想要修改的意圖。...
React——react-redux
因為自己是Vue開發者,所以會以Vuex的角度來看React的狀態管理實現,這樣可以幫助自己理解。畢竟兩者都是參照Flux來分別實現的。之前寫過一篇Vuex的分享 Vue中狀態管理——Vuex 說起 react-redux 就不得不先說 redux。 Redux redux 是一個數據設計架構,專注于狀態管理,和 react 解耦 單一狀態,單向數據流 redux 中有4個...
React React-redux
學習目標 React-Redux學習(不需用subscribe) 安裝: cnpm install react-redux --save 概念: Provider組件:自動的將store里的state和組件進行關聯。 MapStatetoProps:這個函數用于將store的state映射到組件的里props mapdispatchToProps:將store中的dispatch映射...
freemarker + ItextRender 根據模板生成PDF文件
1. 制作模板 2. 獲取模板,并將所獲取的數據加載生成html文件 2. 生成PDF文件 其中由兩個地方需要注意,都是關于獲取文件路徑的問題,由于項目部署的時候是打包成jar包形式,所以在開發過程中時直接安照傳統的獲取方法沒有一點文件,但是當打包后部署,總是出錯。于是參考網上文章,先將文件讀出來到項目的臨時目錄下,然后再按正常方式加載該臨時文件; 還有一個問題至今沒有解決,就是關于生成PDF文件...
猜你喜歡
電腦空間不夠了?教你一個小秒招快速清理 Docker 占用的磁盤空間!
Docker 很占用空間,每當我們運行容器、拉取鏡像、部署應用、構建自己的鏡像時,我們的磁盤空間會被大量占用。 如果你也被這個問題所困擾,咱們就一起看一下 Docker 是如何使用磁盤空間的,以及如何回收。 docker 占用的空間可以通過下面的命令查看: TYPE 列出了docker 使用磁盤的 4 種類型: Images:所有鏡像占用的空間,包括拉取下來的鏡像,和本地構建的。 Con...
requests實現全自動PPT模板
http://www.1ppt.com/moban/ 可以免費的下載PPT模板,當然如果要人工一個個下,還是挺麻煩的,我們可以利用requests輕松下載 訪問這個主頁,我們可以看到下面的樣式 點每一個PPT模板的圖片,我們可以進入到詳細的信息頁面,翻到下面,我們可以看到對應的下載地址 點擊這個下載的按鈕,我們便可以下載對應的PPT壓縮包 那我們就開始做吧 首先,查看網頁的源代碼,我們可以看到每一...
Linux C系統編程-線程互斥鎖(四)
互斥鎖 互斥鎖也是屬于線程之間處理同步互斥方式,有上鎖/解鎖兩種狀態。 互斥鎖函數接口 1)初始化互斥鎖 pthread_mutex_init() man 3 pthread_mutex_init (找不到的情況下首先 sudo apt-get install glibc-doc sudo apt-get install manpages-posix-dev) 動態初始化 int pthread_...
統計學習方法 - 樸素貝葉斯
引入問題:一機器在良好狀態生產合格產品幾率是 90%,在故障狀態生產合格產品幾率是 30%,機器良好的概率是 75%。若一日第一件產品是合格品,那么此日機器良好的概率是多少。 貝葉斯模型 生成模型與判別模型 判別模型,即要判斷這個東西到底是哪一類,也就是要求y,那就用給定的x去預測。 生成模型,是要生成一個模型,那就是誰根據什么生成了模型,誰就是類別y,根據的內容就是x 以上述例子,判斷一個生產出...