-
Notifications
You must be signed in to change notification settings - Fork 547
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
完全理解 redux(从零实现一个 redux) #22
Comments
大神的理解很透彻,通俗易懂,学习了。 |
@Mrzhangyu 感谢认可。谢谢 |
看了这么多redux的教程,这个真的是很通俗易懂了,不过最后中间件的地方还得好好捋一捋,还是有点晕,感谢大神 |
这个讲的通俗易懂,赞-- |
@maojiebaibaibai 中间件那里,建议跟着写一遍。自己实现两个中间件就会了! |
这个写的真的很好,比网上看到很多redux的教程写的好 |
厉害厉害,真的是用心了 |
厉害,讲的很好, |
前端 真难 |
这个真心赞一个。 |
点赞。 |
解释的非常清晰就算初学者也能看懂 感谢作者 |
为什么你这么厉害,真的通俗易懂唉 |
这个是真的大神,实在佩服,十分感谢,希望继续出关于dva实战的文章 |
666,是看过的关于redux最棒的教程 |
啥时候能实现下 |
真是太厉害了,终于明白了种种的种种 |
多文件协作第三段代码 |
赞👍 |
非常棒,赞 |
看的很透彻 感谢大神 |
讲解的通俗易懂,点赞 |
感谢 |
送上我的膝盖 |
通过简述过程中把redux源码解析出来, 大神666 |
牛逼 讲得很透彻 |
功力得多么深厚,才能把这个讲的这么透彻啊 |
果然是大神,这是我看的最清晰的redex叫教程了。感谢!!!! |
很通俗易懂,我先过一遍。项目完成之后,再跟着你这写一遍。谢谢分享~~真的很棒👍 |
真是一种享受,要是多一些这样的文章该多好! |
之前学习过,现在来复习。写的真棒 👍 |
优秀
数组是长度为1时候reduce默认就是第一个元素,这边判断是不是多余了 |
1 similar comment
优秀
数组是长度为1时候reduce默认就是第一个元素,这边判断是不是多余了 |
perfect |
跟着写了一遍,很赞啊,很全面! |
讲得很好,跟着走了一遍,循序渐进的,很好理解 |
我也认同reduce 更好(看到后边发现了),但map是没有必要的,forEach更合适,这里没有必要创建一个无用的新数组 |
大佬讲得非常通透易懂哇!不过总结那里有一句话“dispatch( action ) 触发 action,生成新的 state”,感觉可能会有一丢丢的歧义【可能会有人理解成dispath和reducer一样都可以更改state,但实际上,更改state的地方只有一个,那就是reducer,无论是redux文档的解释还是大佬代码里的表现显示都是如此】。多加几个字,改成“dispatch( action ) 触发 action,通知reducer按计划生成新的 state”,这样子也许更好一点?当然这只是小弟的个人感觉哈 |
大佬,通透👍 |
终于看完了,临摹了一遍,收获很大 |
牛逼 |
谢谢大神,很有帮助! |
写的真好,看过的有关文章里最循序渐进通俗易懂的了 applyMiddleware那里把store抽出来返回函数,参数是action,有点依赖注入和惰性求值的味道,虽然不完全是纯函数 如果中间件那里加上异步处理的解析,比如外部的runSaga(watchers),那就更棒了 |
谢谢! |
这是来自QQ邮箱的自动回复邮件。您好,您的邮件我已收到 ——刘慧
|
unsubscribe的实现方式有问题,以作者给出的DEMO-9,index.js中的代码为例 const un = store.subscribe(() => {
let state = store.getState();
console.log(state.counter.count);
});
store.dispatch({
type: 'INCREMENT'
});
un()
store.dispatch({
type: 'INCREMENT'
});
console.log(store.getState()) // { counter: { count: 3 } } 取消订阅后,count的值仍然发生了变化 |
这是一封自动回复邮件。已经收到您的来信,我会尽快回复。
|
这是来自QQ邮箱的自动回复邮件。您好,您的邮件我已收到 ——刘慧
|
可以的,值得学习 |
这是一封自动回复邮件。已经收到您的来信,我会尽快回复。
|
这是来自QQ邮箱的自动回复邮件。您好,您的邮件我已收到 ——刘慧
|
完全理解 redux
关注公众号「前端技术砖家」,拉你进交流群,大家一起共同交流和进步。
目录
记得开始接触 react 技术栈的时候,最难理解的地方就是 redux。全是新名词:reducer、store、dispatch、middleware 等等,我就理解 state 一个名词。
网上找的 redux 文章,要不有一本书的厚度,要不很玄乎,晦涩难懂,越看越觉得难,越看越怕,信心都没有了!
花了很长时间熟悉 redux,慢慢的发现它其实真的很简单。本章不会把 redux 的各种概念,名词解释一遍,这样和其他教程没有任何区别,没有太大意义。我会带大家从零实现一个完整的 redux,让大家知其然,知其所以然。
开始前,你必须知道一些事情:
Let's Go!
简单的状态管理器
redux 是一个状态管理器,那什么是状态呢?状态就是数据,比如计数器中的 count。
我们来使用下状态
我们来修改下状态
好了,现在我们实现了状态(计数)的修改和使用了。
当然上面的有一个很明显的问题:修改 count 之后,使用 count 的地方不能收到通知。我们可以使用发布-订阅模式来解决这个问题。
我们来尝试使用下这个简单的计数状态管理器。
现在我们可以看到,我们修改 count 的时候,会输出相应的 count 值。
现在有两个新的问题摆在我们面前
我们尝试来解决这个问题,把公共的代码封装起来
我们来使用这个状态管理器管理多个状态 counter 和 info 试试
到这里我们完成了一个简单的状态管理器。
这里需要理解的是
createStore
,提供了changeState
,getState
,subscribe
三个能力。本小节完整源码见 demo-1
有计划的状态管理器
我们用上面的状态管理器来实现一个自增,自减的计数器。
你一定发现了问题,count 被改成了字符串
abc
,因为我们对 count 的修改没有任何约束,任何地方,任何人都可以修改。我们需要约束,不允许计划外的 count 修改,我们只允许 count 自增和自减两种改变方式!
那我们分两步来解决这个问题
我们来设置一个 plan 函数,接收现在的 state,和一个 action,返回经过改变后的新的 state。
我们把这个计划告诉 store,store.changeState 以后改变 state 要按照我的计划来改。
我们来尝试使用下新的 createStore 来实现自增和自减
到这里为止,我们已经实现了一个有计划的状态管理器!
我们商量一下吧?我们给 plan 和 changeState 改下名字好不好?**plan 改成 reducer,changeState 改成 dispatch!**不管你同不同意,我都要换,因为新名字比较厉害(其实因为 redux 是这么叫的)!
本小节完整源码见 demo-2
reducer 的拆分和合并
这一小节我们来处理下 reducer 的问题。啥问题?
我们知道 reducer 是一个计划函数,接收老的 state,按计划返回新的 state。那我们项目中,有大量的 state,每个 state 都需要计划函数,如果全部写在一起会是啥样子呢?
所有的计划写在一个 reducer 函数里面,会导致 reducer 函数及其庞大复杂。按经验来说,我们肯定会按组件维度来拆分出很多个 reducer 函数,然后通过一个函数来把他们合并起来。
我们来管理两个 state,一个 counter,一个 info。
他们各自的 reducer
那我们用 combineReducers 函数来把多个 reducer 函数合并成一个 reducer 函数。大概这样用
我们尝试实现下 combineReducers 函数
我们来尝试下 combineReducers 的威力吧
本小节完整源码见 demo-3
state 的拆分和合并
上一小节,我们把 reducer 按组件维度拆分了,通过 combineReducers 合并了起来。但是还有个问题, state 我们还是写在一起的,这样会造成 state 树很庞大,不直观,很难维护。我们需要拆分,一个 state,一个 reducer 写一块。
这一小节比较简单,我就不卖关子了,用法大概是这样(注意注释)
我们修改下 createStore 函数,增加一行
dispatch({ type: Symbol() })
我们思考下这行可以带来什么效果?
state = reducer(state, action)
你可以试试
本小节完整源码见 demo-4
到这里为止,我们已经实现了一个七七八八的 redux 啦!
中间件 middleware 是 redux 中最难理解的地方。但是我挑战一下用最通俗的语言来讲明白它。如果你看完这一小节,还没明白中间件是什么,不知道如何写一个中间件,那就是我的锅了!
中间件是对 dispatch 的扩展,或者说重写,增强 dispatch 的功能!
记录日志
我现在有一个需求,在每次修改 state 的时候,记录下来 修改前的 state ,为什么修改了,以及修改后的 state。我们可以通过重写 store.dispatch 来实现,直接看代码
我们来使用下
日志输出为
现在我们已经实现了一个完美的记录 state 修改日志的功能!
记录异常
我又有一个需求,需要记录每次数据出错的原因,我们扩展下 dispatch
这样每次 dispatch 出异常的时候,我们都会记录下来。
多中间件的合作
我现在既需要记录日志,又需要记录异常,怎么办?当然很简单了,两个函数合起来呗!
如果又来一个需求怎么办?接着改 dispatch 函数?那再来10个需求呢?到时候 dispatch 函数肯定庞大混乱到无法维护了!这个方式不可取呀!
我们需要考虑如何实现扩展性很强的多中间件合作模式。
我们把 loggerMiddleware 提取出来
我们把 exceptionMiddleware 提取出来
现在的代码有一个很严重的问题,就是 exceptionMiddleware 里面写死了 loggerMiddleware,我们需要让
next(action)
变成动态的,随便哪个中间件都可以同样的道理,loggerMiddleware 里面的 next 现在恒等于 store.dispatch,导致 loggerMiddleware 里面无法扩展别的中间件了!我们也把 next 写成动态的
到这里为止,我们已经探索出了一个扩展性很高的中间件合作模式!
这时候我们开开心心的新建了一个
loggerMiddleware.js
,一个exceptionMiddleware.js
文件,想把两个中间件独立到单独的文件中去。会碰到什么问题吗?loggerMiddleware 中包含了外部变量 store,导致我们无法把中间件独立出去。那我们把 store 也作为一个参数传进去好了~
到这里为止,我们真正的实现了两个可以独立的中间件啦!
现在我有一个需求,在打印日志之前输出当前的时间戳。用中间件来实现!
本小节完整源码见 demo-6
中间件使用方式优化
上一节我们已经完全实现了正确的中间件!但是中间件的使用方式不是很友好
其实我们只需要知道三个中间件,剩下的细节都可以封装起来!我们通过扩展 createStore 来实现!
先来看看期望的用法
实现 applyMiddleware
让用户体验美好
现在还有个小问题,我们有两种 createStore 了
为了让用户用起来统一一些,我们可以很简单的使他们的使用方式一致,我们修改下 createStore 方法
最终的用法
本小节完整源码见 demo-7
退订
不能退订的订阅都是耍流浪!我们修改下 store.subscribe 方法,增加退订功能
使用
中间件拿到的store
现在的中间件拿到了完整的 store,他甚至可以修改我们的 subscribe 方法,按照最小开放策略,我们只用把 getState 给中间件就可以了!因为我们只允许你用 getState 方法!
修改下 applyMiddleware 中给中间件传的 store
compose
我们的 applyMiddleware 中,把 [A, B, C] 转换成 A(B(C(next))),是这样实现的
redux 提供了一个 compose 方式,可以帮我们做这个事情
看下他是如何实现的
当然 compose 函数对于新人来说可能比较难理解,你只需要他是做什么的就行啦!
省略initState
有时候我们创建 store 的时候不传 initState,我们怎么用?
redux 允许我们这样写
我们仅需要改下 createStore 函数,如果第二个参数是一个object,我们认为他是 initState,如果是 function,我们就认为他是 rewriteCreateStoreFunc。
2 行代码的 replaceReducer
reducer 拆分后,和组件是一一对应的。我们就希望在做按需加载的时候,reducer也可以跟着组件在必要的时候再加载,然后用新的 reducer 替换老的 reducer。
我们来尝试使用下
replaceReducer 示例源码见 demo-5
bindActionCreators
bindActionCreators 我们很少很少用到,一般只有在 react-redux 的 connect 实现中用到。
他是做什么的?他通过闭包,把 dispatch 和 actionCreator 隐藏起来,让其他地方感知不到 redux 的存在。
我们通过普通的方式来 隐藏 dispatch 和 actionCreator 试试,注意最后两行代码
我眼睛一看,这个 actions 生成的时候,好多公共代码,提取一下
来看一下 bindActionCreators 的源码,超级简单(就是生成了刚才的 actions)
bindActionCreators 示例源码见 demo-8
大功告成
完整的示例源码见 demo-9,你可以和 redux 源码做一下对比,你会发现,我们已经实现了 redux 所有的功能了。
当然,为了保证代码的理解性,我们少了一些参数验证。比如
createStore(reducer)
的参数 reducer 必须是 function 等等。纯函数
什么是纯函数?
纯函数是这样一种函数,即相同的输入,永远会得到相同的输出,而且没有任何可观察的副作用。
通俗来讲,就两个要素
我们的 reducer 计划函数,就必须是一个纯函数!
只要传入参数相同,返回计算得到的下一个 state 就一定相同。没有特殊情况、没有副作用,没有 API 请求、没有变量修改,单纯执行计算。
到了最后,我想把 redux 中关键的名词列出来,你每个都知道是干啥的吗?
createStore
创建 store 对象,包含 getState, dispatch, subscribe, replaceReducer
reducer
reducer 是一个计划函数,接收旧的 state 和 action,生成新的 state
action
action 是一个对象,必须包含 type 字段
dispatch
dispatch( action )
触发 action,生成新的 statesubscribe
实现订阅功能,每次触发 dispatch 的时候,会执行订阅函数
combineReducers
多 reducer 合并成一个 reducer
replaceReducer
替换 reducer 函数
middleware
扩展 dispatch 函数!
你再看 redux 流程图,是不是大彻大悟了?
(redux 流程图)
The text was updated successfully, but these errors were encountered: