第 62 题:redux 为什么要把 reducer 设计成纯函数?

发布于 2022-08-22 06:49:48 字数 105 浏览 119 评论 10

什么是 reducer?之所以将这样的函数称之为 reducer,是因为这种函数与被传入 Array.prototype.reduce(reducer, ?initialValue) 的回调函数属于相同的类型。

如果你对这篇内容有疑问,欢迎到本站社区发帖提问 参与讨论,获取更多帮助,或者扫码二维码加入 Web 技术交流群。

扫码二维码加入Web技术交流群

发布评论

需要 登录 才能够评论, 你可以免费 注册 一个本站的账号。

评论(10

夏九 2022-05-04 13:09:33

reducer 规定为纯函数, 个人认为主要有两点:
其一是保证新旧 state 不是同一个对象引用, 所以不能直接修改旧 state 中的数据, 然后返回旧 state, 这会导致 combineReducers 中的判断失效( hasChanged = hasChanged || nextStateForKey !== previousStateForKey ), 最后redux会直接返回旧 state, 从而导致操作无效;
其二如楼上 yyyddc 所说, 为了保证返回的新 state 是确定的, 而不是会因为某些副作用返回不确定值的 state.

热情消退 2022-05-04 12:57:01

可以把redux改一改,不要action、reducer,一样能让你的react响应redux的状态的变化。但是没有reducer的redux就失去了‘状态管理’的功能了,你看不到状态的变跟流,弄不了所谓的‘时间回溯’。为什么框架让你整纯函数,就是要你保证状态的变跟的确定性。异步是没发保证的,先后顺序、值都没法百分白保证,不确定的东西就影响了这个框架所谓的‘基石’;明确的状态变更流、引以为傲的设计初衷,你不写成纯函数可以吗?完全ojbk的,但是这不是好的解决思路。你可以reducer和effect分开,如果你要个参考,dva?或者更直接点,vuex的模式是我个人认为最好的解决方案,effect和reducer在书写结构层面就分离,清晰明了

醉殇 2022-05-04 11:57:16
  1. redux里会订阅重新渲染事件,这样数据改变会重渲染。
  2. 当我们dispatch一个action时,reducer会将(state和action)作为参数依赖生成一个全新的state,替代redux里的state,并且触发订阅的事件重新渲染(即已知纯函数,返回唯一,且无副作用)
  3. 如果reducer带副作用,会是什么样?修改redux内部的state?
  4. 正如react有一个shouldcomponentupdate事件作为优化,当我们重渲染时也希望订阅事件能差异化更新以优化性能,怎么实现?比对一次更新里的新老state
  5. 当reducer是纯函数时,生成一个新state,新老对象地址引用不一,能实现
    若reducer是副作用,那么虽然对象内变化,新老对象地址一致,不能实现
  6. 以上观点纯属是看了一些小文章后的个人观点。感觉对象的深比较也能实现。个人觉得函数式编程本身就是react推崇的一大范式。
仙女山的月亮 2022-05-04 10:38:27

可以倒退答案
纯函数的作用是没有副作用,唯一输入对应唯一输出;然后看下redux的作用,状态管理工具,为react等前端框架管理数据流。数据就是输入,react视图状态就是输出。唯一输入必须对应唯一的输出,不然数据和视图就对不上了。
所以redux必须是纯函数。

烟花易冷人易散 2022-05-04 10:04:08

首先命题应当改一下,中文有歧义,可能改为 “redux中的reducer为什么必须(最好)是纯函数“,我想表达的意思是,redux没有强制你reducer是个纯函数,事实上,没有人能通过框架限制判断一个函数是否是纯函数,所以题目中的'设计成'这个短语貌似在说redux已经把reducer强制规定是纯函数了。这回让你怀疑你对redux的认知。

正文如下

然后说一下为什么reducer最好是纯函数,首先你得看看文档怎么说reducer的作用的,‘接收旧的 state 和 action,返回新的 state’,您可得瞧好咯,他就是起一个对数据做简单处理后返回state的作用,为什么只起这个作用,这时用设计这个词回答这个问题才恰当,因为redux把reducer设计成只负责这个作用。很白痴的问答对吧,所以题目的答案也就简单了,reducer的职责不允许有副作用,副作用简单来说就是不确定性,如果reducer有副作用,那么返回的state就不确定,举个例子,你的reducer就做了一个value = value + 1这个逻辑,然后返回state为{value},ok,这个过程太jr纯了,然后你可能觉得要加个请求来取得value后再加1,那么你的逻辑就是value = getValue() + 1, getValue是个请求函数,返回一个值,这种情况,退一万步讲,如果你的网络请求这次出错,那么getValue就返回的不是一个数值,value就不确定了,所以return的state你也不确定了,前端UI拿到的数据也不确定了,所以就是这个环节引入了副作用,他娘的redux设计好的规范就被你破坏了,redux就没卵用了。到此为止这个问题回答完了,我没有说什么上面几个jr说的教科书的理论,甚至还加了些脏话。请原谅,这只是戏剧需要。

最后我回答下如何解决这个副作用,实际上也很白痴的问题,这里的请求可以放在reducer之前,你先请求,该做出错处理的就做出错处理,等拿到实际数据后在发送action来调用reducer。这样通过前移副作用的方式,使reducer变得纯洁。

隔岸观火 2022-05-04 08:35:31

store 里的 state 是一个引用类型,多个组件都可能共享这个 state,如果允许直接在组件中修改这个 state,由于组件间千丝万缕的关系,复杂度会变得很高,定位问题会变得异常困难,因为很难搞清楚到底是哪个组件“搞坏”了数据,而采用纯函数就没有这样的副作用。

表情可笑 2022-05-03 22:04:41

redux三大原则

  1. 单一数据流
    整个应用state都被储存在一个store里面 构成一个Object tree
  2. State是只读的
    唯一改变state的方法就是触发action, action是一个用于描述已发生事件的普通对象
  3. 使用纯函数来执行修改
    为了描述action如何改变state tree, 你需要编写reducers

把reducer设计成纯函数,可以实现时间旅行,记录/回放或者热加载

提笔书几行 2022-05-02 14:25:50

redux的设计思想就是不产生副作用,数据更改的状态可回溯,所以redux中处处都是纯函数

函数式编程思想?

层林尽染 2022-05-02 11:02:01

redux的设计思想就是不产生副作用,数据更改的状态可回溯,所以redux中处处都是纯函数

~没有更多了~

关于作者

软甜啾

暂无简介

0 文章
0 评论
22 人气
更多

推荐作者

lorenzathorton8

文章 0 评论 0

Zero

文章 0 评论 0

萧瑟寒风

文章 0 评论 0

mylayout

文章 0 评论 0

tkewei

文章 0 评论 0

17818769742

文章 0 评论 0

    我们使用 Cookies 和其他技术来定制您的体验包括您的登录状态等。通过阅读我们的 隐私政策 了解更多相关信息。 单击 接受 或继续使用网站,即表示您同意使用 Cookies 和您的相关数据。
    原文