Redux学习之二:从action到store?

前端之家收集整理的这篇文章主要介绍了Redux学习之二:从action到store?前端之家小编觉得挺不错的,现在分享给大家,也给大家做个参考。

redux里面的概念很多,有Action、ActionCreator、reducer、store、middleware、Provider、Connect……概念不理解,就是眉毛胡子一把抓,弄不出头绪。redux的概念,通过一张图大家就一目了然了。

这张图大致可以概括redux的整个流程。从图中我们可以看出,Action是数据流动的开始,Reducer负责业务逻辑处理,store是整个流程的中轴。

1、你从哪里来?——谈谈actions的起源

redux里的action和flux里的action大体上一致,不同的是设计理念的差别。flux里的action是以函数的形式,但在redux里,action被设计成普通的js对象。

{
type: 'ADD_TODO',
text: 'Build my first Redux app'
}
这就是一个最基本的action,必须注意的是,type是action必须的一个key值,是action的唯一标识。其它的key可以使任意像传递的数据结构。
但是actions到底代表什么?
actions代表着数据来源的萃取。我们的view界面需要很多数据结构,如ajax返回的数据、路由状态跟踪、UI状态等。。。关键的是这些数据都是动态流动的,既然流动,就要有入口、有方向。actions就是代表数据流动的开始,携带的key值中,type代表发生了什么事情,其它的就是需要流动的数据结构。
归纳起来,actions是数据从应用传到store的有效载荷,是store唯一的数据来源。
那么如何产生一个action,这就是又有一个概念actionCreator:

function addTodo(text) {

  1. return {
  2. type: ADD_TODO,text
  3. };

}
actionCreator实际上是一个返回action值的函数而已。这样,只需把 action 生成器的结果传给 dispatch() 方法即可实例化 dispatch。

  1. dispatch(addTodo(text));

2、条条大路通哪里?——看看reducer的神奇

action表示发生了什么事情,表明了有事情发生,肯定会对应用的数据产生影响。我们知道,React组件内的数据都是以state的形式存在的,state代表着数据结构。那么就要问了,action的发生到底会对state产生什么影响。这就是reducer的作用了。
reducer 其实是一个函数, 接收旧的 state 和 action, 返回新的 state。

  1. (prevIoUsState,action) => newState

这种设计来源于函数式编程思想,简单,易懂,没有副作用。

function todoApp(state = initialState,action) {
switch (action.type) {
case SET_VISIBILITY_FILTER:

  1. return Object.assign({},state,{
  2. visibilityFilter: action.filter
  3. });

default:

  1. return state;

}
}
可以看出,业务逻辑部分都是在reducer里处理。这就涉及一个问题,一个应用里有很多逻辑部分,都放在一个reducer里面会造成非常的臃肿。实际过程中,把业务进行分拆,然后通过combineReducer函数合并成一个reducer。详细可以看这里:
https://github.com/rackt/redux/blob/mast...
切记,在reducer不要改变旧的state值。

3、看看你是谁?——揭开store的面纱

首先我们看文档里怎么描述的:

The Store is the object that brings themtogether.(them指的是action和reducer)

It’s important to note that you’ll only have a single store in a Redux application.

这就说明,Store负责把reducer和action结合的作用。store怎么创建?一般是通过下面的代码

const store = createStore(reducer);
这个createStore又是什么函数,我们看看createStore.js源码:

  1. import isPlainObject from './utils/isPlainObject';
  2. export var ActionTypes = {
  3. INIT: '@@redux/INIT'
  4. };
  5. export default function createStore(reducer,initialState) {
  6. if (typeof reducer !== 'function') {
  7. throw new Error('Expected the reducer to be a function.');
  8. }
  9.  
  10. var currentReducer = reducer;
  11. var currentState = initialState;
  12. var listeners = [];
  13. var isDispatching = false;
  14.  
  15. function getState() {
  16. return currentState;
  17. }
  18.  
  19. function subscribe(listener) {
  20. listeners.push(listener);
  21.  
  22. return function unsubscribe() {
  23. var index = listeners.indexOf(listener);
  24. listeners.splice(index,1);
  25. };
  26. }
  27.  
  28. function dispatch(action) {
  29. if (!isPlainObject(action)) {
  30. throw new Error(
  31. 'Actions must be plain objects. ' +
  32. 'Use custom middleware for async actions.'
  33. );
  34. }
  35.  
  36. if (typeof action.type === 'undefined') {
  37. throw new Error(
  38. 'Actions may not have an undefined "type" property. ' +
  39. 'Have you misspelled a constant?'
  40. );
  41. }
  42.  
  43. if (isDispatching) {
  44. throw new Error('Reducers may not dispatch actions.');
  45. }
  46.  
  47. try {
  48. isDispatching = true;
  49. currentState = currentReducer(currentState,action);
  50. } finally {
  51. isDispatching = false;
  52. }
  53.  
  54. listeners.slice().forEach(listener => listener());
  55. return action;
  56. }
  57.  
  58. function replaceReducer(nextReducer) {
  59. currentReducer = nextReducer;
  60. dispatch({ type: ActionTypes.INIT });
  61. }
  62.  
  63.  
  64. dispatch({ type: ActionTypes.INIT });
  65.  
  66. return {
  67. dispatch,subscribe,getState,replaceReducer
  68. };
  69. }

我们可以看到createStore返回的是一个对象,该对象有四种方法,分别是:

  1. dispatch,replaceReducer

可以看出redux里的store有点像flux里的dispatcher的作用,产生action可以通过store.dispatch发送出去,想监听状态可以通过store.subscribe订阅
值得注意的是,程序初始化的时候,redux会发送一个类型为@@redux/INIT的action,来初始化state。

以上就是我对redux流程前半部分的理解,请批评指正。

猜你在找的React相关文章