Redux 现在真的已经变成了前端 react 项目的标配了,但看了很多同学写的代码,大家的用法并不一致,甚至根本没有用到 redux 的本质的东西,下面我希望跟大家讨论下 Redux 到底该怎么用?以及为什么我觉得不应该在复杂的业务开发中使用 redux。
这里说的怎么用并不说,他的写法,action、reducer、store 之间如何调用,而是说他的理念,Redux 到底希望你什么情况下使用他,又该怎么用好他。
Redux 只是一个状态机吗?
有多少人,在用 Redux 之前,是把他的官方文档详细的看过一遍的呢?有多少人,是看了一些二手的文章中的例子,就去写代码的。
官方说:「Redux 是 JavaScript 应用的状态容器,提供可预测的状态管理」。简单易懂,Redux 就是一个状态机嘛。从本质上来说,其实并没有错,但官方的解释并没有立足前端的场景,Redux 如果只是作为一个状态机,他可以出现在任何使用 JavaScript 写的代码里,用 NodeJS 写一个后端程序也可以用 Redux 维护一个状态机。因此状态机是他的本质,但在前端场景,如果你只是把他当做一个状态机,那就把他看小了。
下面的讨论,我们需要站在前端项目的视角,我们以 Redux + redux-saga 为例,看看 Redux 到底在解决前端的什么问题。
Redux 在解决前端的什么问题?
在没有 Redux 的时代,我们使用 React 编写代码,常常会遇到下面的问题。
-
状态管理混乱:所有的状态分布在不同的位置,难以获取到一个页面完整的数据结构
-
组件层级较深时,跨组件的数据传递需要层层 Props 传递
-
一个数据的改变可能会带来一连串组件的更新
-
数据的改变不能被监听,需要 console.log 看变化,费时费力
-
单页应用中,存在全局状态需要管理,且多个页面依赖这个数据,且有可能会改变全局数据
-
组件数据、行为、和 UI 耦合在一起。
接下来我们看看 Redux 是怎么去解决这些问题的?
1、统一 store ,集中存储页面所有数据,在任何一个页面都可以获取全部数据,也可以做更新操作。
import { createStore } from 'redux';
import createSagaMiddleware from 'redux-saga';
import reducer from './reducers'
import mySaga from './sagas'
// 注入 saga 中间件
const sagaMiddleware = createSagaMiddleware()
// 初始化一个 store
const store = createStore(
reducer,
applyMiddleware(sagaMiddleware)
)
// 运行 saga
sagaMiddleware.run(mySaga)
2、redux 将组件,数据,行为三者分离,组件全是纯 UI 组件,reducer 负责处理状态变化,saga 负责处理请求
这里要注意 redux 其实只关心数据 state ,对于组件的 UI 的层面,是需要 react-redux 来解决的。因此 redux 并不是天生为 react 而生的。
import React from 'react';
import { connect } from 'react-redux';
// UI 组件
const App = ({
value,
onChange,
onSubmit,
}) => {
return (<div>
<input onChange={onChange} value={value} />
<button onClick={onSubmit}>提交</button>
</div>)
};
// react-redux 包装 UI 组件
function mapStateToProps(state) {
return {
value: state.value
}
};
function mapDispatchToProps(dispatch) {
return {
onChange: () => dispatch({ type: 'app/updateValue', payload: value }),
onSubmit: () => dispatch('app/submitForm'),
}
};
const AppContanier = connect(
mapStateToProps,
mapDispatchToProps
)(App);
// 行为
import { call } from 'redux-saga/effects'
// 使用 saga 对 请求 副作用行为的处理
function* submitForm() {
const res = yield call(window.fetch, '/api/submit');
if (res) {
alert('提交成功');
}
}
// reducer 更新数据
function reducer(state = { value: '卡牛' }, action) {
switch (action.type) {
case 'updateValue':
return { value: action.payload }
default:
return state
}
}
3、跨组件传值,使用 connect ,可以避免跨组件传递 props 从而去除不必要的渲染。
4、由于所有的数据改变都要通过 action 触发,因此可以通过 redux-dev-tools 的浏览器插件监听变化,并很好的显示出来,调试效率一流。
你真的用好 Redux 了吗?
Redux 的理念来自于 Facebook 用于构建客户端Web应用程序的基本架构思想 Flux。因此 Redux 核心想向使用者传达的是就是一种编码的思想,但是思想的东西并没有什么硬约束,由于这种灵活性,带来一个比较大的问题就是,每个人的代码写的并不一致。我先来说说,我在 Codereview 别人代码的时候,经常看到的几个问题。
1、不写纯 UI 组件
虽然已经在用 Redux 了,但 React 组件里依然有较多的 state ,这些 state 不只是像 visible 这种简单的状态,还有很多业务数据。更有甚者会出现很多派生 state,这导致数据管理变得异常复杂,尤其是在一次一次迭代以后,堆积的需求代码将持续恶化这种现象。最终导致重构极为复杂。
2、一个页面里只有根组件使用了 connect ,其他子、孙组件都没有使用,而且子、孙组件也并不是简单的 UI 组件,都是有比较多 state 状态的,尤其是有些公共组件里还有接口请求。
3、父子孙组件之间,依然在传递 props 数据,更有甚者直接使用 {...this.props} 传递数据。这种做法完全没有用到 Redux 在跨组件传递数据时的优势。
<FatherComp>
<SonComp {...this.props}>
<GrandsonComp {this.props} />
</SonComp>
</FatherComp>
业务代码里别再用 redux 了!
说了这么多,目的是引出我的观点,我的观点是在业务代码里不要用 redux。我有两方面的论据,来佐证我的观点。
方面一:Redux 能解决的问题,React 都能解决
上面提到的那些问题,真的在业务代码开发中,我觉得只有两个问题值得我们深入关注。
① 跨组件传值,并减少不必要的刷新。
② 跨页面共享数据。
然而,这两个问题也在渐渐消失,这归功于 hooks 的出现,尤其是 useContext 的 API。跨组件传值自不必说,你可以在使用 Context.Provider 包裹下的所有组件内部调用 useContext 获取到全局的数据。
至于减少不必要的刷新,这里确实没有 redux 方便,使用 hooks 会比较麻烦,大概有以下集中方法:
-
使用 React.memo 包裹组件
-
使用 useMemo 包裹 renderXXX 方法
-
拆分 Context
-
使用第三方 npm 包,react-tracked
确实,使用 hooks 的方式解决多次刷新的问题比较麻烦,但结合我的实践经验,如果你的组件拆分力度合理,可以手动控制刷新频率,这样你对于自己代码的了解程度会更高,一些 bug 也更容易得到解决。
方面二:Redux 带来的问题,比他能解决的问题还要多
真实的开发中,如果遇到项目复杂,多人开发,需求井喷,快速迭代的情况的下,使用 redux 会带来非常大的问题。
-
耦合:写代码提倡高内聚低耦合,可由于 redux 的存在,我们把整个项目的数据存储都放在了一起,或者以页面为维度,这种情况其实是违背这一原则的。因为每个页面都有自己独有的业务数据,甚至每个组件都有自己独立的数据,现在把这些数据统一管理,相当于在数据层面做到了耦合,且耦合了很多完全不相干的数据。组件自己管自己的行为和数据,才是真正业务开发中的王道,他极大的隔离了风险,不会污染别的组件。在真实的开发中,业务组件(带请求)的组件才是更多的,UI 组件有了 antd 这种组件库以后,其实很少见了。redux 提倡我们把数据都放到外部 store 里,把组件自身搞成一个 UI 组件,是很不实事求是的。
-
难以抽组件:在写需求的时候,会遇到一个组件本来只有一个地方用,后来的需求也要用到他,甚至其他项目也要用,这个时候我们就会把它抽出来,做成一个 npm 包,但是用 redux 写的组件该怎么写成 npm 包呢?难道要把一个 redux 库引到一个 npm 包里吗?
-
难以重构:我工作中经常接手别人的代码,如果遇到的项目是用 redux 或者 dva 写的,我就想祝你健康。这样的代码是真的很难拆,一个组件 1000 多行,数据有在 state 里的,有在 store 里的,且数据都是通过 props 透传的,咱拆呢?重写又不敢。
-
难以写出好代码:如果一个框架,他有很好的思想,但是需要你在用的时候注意这,注意那。这些思想没有在代码层面做限制,而只是停留在文档里,那这绝对不是一个好的框架。使用 redux 就有这种感觉的,他的数据流思想是很妙的,有一套吸引人的架构范式,但只停留在了思想上,而且这种思想接受门槛也不低。程序员大家的水平真的参差不齐,有多少人真正理解了吗?或者说理解了然后在代码里实践了最佳规范了吗?这个行业很多人真的只是码农,为了完成需求无恶不做那种,完成才是第一要务,至于写好,那就得以后我有空了再重构吧。可等你有空的时候,项目都不一定是你来做了。
-
模板代码太多:思想是好思想,实现是真的丑陋。看着这些差不多的模板代码,作为一个以复用为根本原则的程序员,你不难受吗?而且竟然有 dispatch 这种东西,type 还是一个字符串,完全失去了程序之间显式的调用关系,查找代码极为费劲。而且在代码组织上,redux 的部分被打平放到了根目录下,在调试的时候,跳来跳去,两个字:难受!
-
不敬畏官方:没有那金刚钻,别揽那瓷器活,redux 官方文档里提到了他适用场景,我读下来就是最好不用的意思,可你把他理解为了都得用,那就是你的不对了。虽然官方也说在「多交互、多数据源」的场景下比较适合使用 redux ,可我实践下来,他忽略了一点,就是一个复杂项目不是一个人来架构和开发的,我们无法保证大家都领会了第一个写代码人的思想。而且需求是在变化的,迭代层出不穷,不仅要考虑加代码,还要考虑改代码,即使不改,也要保证删的方便。
总结
写这篇文章,多多少少是带着些情绪的,因为我确实在这一年多的时间里,重构了大量基于 Redux 写的代码,每个都是几万甚至十几万行代码的 Git 库。1000 行以上的组件也很多,我深刻体会到这其中的难处。相比而言,Hooks 的好处则跃然纸上。我所做的中后台项目,每个页面都很复杂,表单众多,一个页面上不同的区块都是不同的业务,交互也很复杂。这种的业务在 Redux 的加持下,简直如同判了死刑,上了刑场,我是想动又不敢动,只能硬着头皮一行一行,拆、改、灰度发布。这个过程是极为痛苦的。当然这种痛苦,可能不完全是 Redux 的锅,是使用者有问题,但不可否认,Redux 起到了助纣为虐的作用。希望有病友能够理解,早点弃坑,少掉点头发。