React应用状态管理(一):理解Action、Reducer、Store

因为正在开发的是一个小项目,除了登录用户信息、全局配置没有其他需要集中管理的状态,就没打算用Redux。杀鸡焉用牛刀,也有尽量少依赖第三方的强迫症,不想整合redux,react-redux和redux-thunk三个新包。当然Redux有很多优点:已经很完善且生态强大、拥有清晰的规范体系和特定的设计约束、以及精挑细选的API。遵循这种状态管理方案的最佳实践,在大型团队项目中可以很好地帮助我们约定代码规则,减少成员出错和乱套的概率。Mobx则是面向对象思维的状态管理库,我还没有用过。
这次我想试试靠React单打,一通百度有样学样用React Hook的useContext结合useReducer模拟了Redux的状态管理机制。实质就是一个可以处理同步/异步动作的函数,并根据动作函数更新状态的集中状态管理器。这边记录下摸索过程的心得体会。

我决定又悄咪咪分个上下两篇:占坑还没写完orz React应用状态管理(二):用React Hook模拟实现Redux

参考文章地址:使用Hooks代替ReduxReact Hooks 版实现 Redux

状态管理三剑客:Action、Reducer、Store(不局限于Redux)

1. Action

Action 就是一个普通JS对象,作用除了描述对 store 数据的更改操作,更是一个把数据从应用传到store的可靠载体(数据有可能是服务器响应,用户输入等等)。它必须包含一个type字段表明要操作的类型。
一般都是通过 store.dispatch() 把 action 传到 store。

const ADD = 'ITEMS_ADD'

{ type: ADD,  payload: 'text' }

在大型项目中,把type的值显式地定义成常量会更利于团队协作。

Action 只做简单的参数传递和配置,最多做一些数据格式的转换,如把 date 转成 formatted string。尽量把数据处理逻辑放在reducer中。因为按照语意 action 就只是一个 identifier,reducer 才是负责逻辑处理的。

Action 创建函数是生成 action 的方法,这样做更方便我们传递数据。
就是像这样简单的返回一个 action:

export function completeItem(id: number) {
  return {
    type: ITEMS.COMPLETE,
    payload: id
  };
}

然后在项目中把 action 创建函数的结果传给 dispatch() 方法即可发起一次 dispatch 过程。dispatch(completeItem(id))
或者创建一个 被绑定的 action 创建函数 来自动 dispatch,再直接调用:
const boundCompleteItem = id => dispatch(completeItem(id))
boundCompleteItem(id);

2. Reducer

reducer 是一个接收旧 state 和 action,并返回新的 state 的函数。它指定了如何响应 actions 并发送到 store 。记住 actions 只是描述了有事情发生了这一事实,并没有描述应用如何更新 state。
它与被传入 Array.prototype.reduce(reducer, ?initialValue) 里的回调函数属于相同的类型,这就是reducer命名的由来。

/** arr.reduce(callback, [initialValue]) **/
/* 第一个参数是回调函数: (prev, curr) => prev + curr * 
与redux当中的reducer模型 (previousState, action) => newState 是不是非常相似呢 */
var sum = [0,1,2,3,4].reduce((prev,curr) => prev + curr); // sum = 10

保持 reducer 纯净非常重要。永远不要在 reducer 里调用非纯函数或有执行副作用的操作(如 API 请求和路由跳转)。
不要直接修改 state 可以用对象展开运算符{ ...state, ...newState } 达到合并新旧数据并返回新指针对象的目的。

一个reducer函数示例

export const initialState = [];

// reducer接收 旧state和 action 并返回新的state
export default function items(state = initialState, action) {
  // 根据不同的action.type对state进行不同的操作
  switch (action.type) {
    case ITEMS.RESET: {
      return [];
    }
    case ITEMS.ADD: {
      return [
        ...state,
        {
          id: Date.now(),
          text: action.payload,
          completed: false,
        },
      ];
    }
    case ITEMS.COMPLETE: {
      return state.map((item) => {
        if (item.id === action.payload) {
          return { ...item, completed: !item.completed }
        }
        return item;
      });
    }
    default: {
      return state;
    }
  }
}

3. Store

Store 只是有几个方法的JS对象。它把Action和Reducer联系到一起,用来维持应用所有的 state 树 。 要创建一个store,只需要把根部的 reducer 函数传递给 createStore。 改变 store 内 state 的惟一途径是对它 dispatch 一个 action
它提供的几个主要方法:

以下是对createStore源码的模拟:

const createStore = (reducer: any)=>{
  let state: any; 
  let listeners:[] = [];
    
  // 用来返回当前的state
  const getState = () => state;

  // 根据action调用reducer返回新的state并触发listener
  const dispatch=(action: any)=>{
    state=reducer(state,action);
    listeners.forEach((listener: any) => listener());
  };
  /* 这里的subscribe有两个功能  
  * 调用 subscribe(listener) 会使用listeners.push(listener)注册一个listener  * 而调用 subscribe 的返回函数则会注销掉listener  */
  const subscribe = (listener: never) => {
    listeners.push(listener);
    return () => {
      listeners:[] = listeners.filter(l=>l!==listener);
    };
  };
  return{ getState, dispatch, subscribe };
};

创建store

import rootReducer from './reducers'
let store = createStore(rootReducer)

最后贴一个用js简单实现的基础版redux。

可以帮助我们捋清redux最基本的流程。异步代码非原创,出处: 浅谈Redux Action & Reducer & Store
下篇接着讲包含异步Action的React Hook版实现。

const store = {
  state: {}, // 全局唯一的state,内部变量,通过getState()获取
  listeners: [], // listeners,用来诸如视图更新的操作
  dispatch: () => {}, // 分发action
  subscribe: () => {}, // 用来订阅state变化
  getState: () => {}, // 获取state
}

const createStore = (reducer, initialState) => {
  // internal variables
  const store = {};
  store.state = initialState;
  store.listeners = [];
  
  // api-subscribe
  store.subscribe = (listener) => {
    store.listeners.push(listener);
    return () => {
      store.listener = store.listeners.filter(l => l !== listener);
    };
  };
  // api-dispatch
  // 根据 action 调用reducer返回新的state,并触发listener
  store.dispatch = (action) => { 
    store.state = reducer(store.state, action);
    store.listeners.forEach(listener => listener());
  };
  
  // api-getState
  store.getState = () => store.state;
  
  return store;
};

// reducer
function counter(state = 0, action) {
  switch (action.type) {
  case 'INCREMENT':
    return state + 1
  case 'DECREMENT':
    return state - 1
  default:
    return state
  }
}

let store = createStore(counter)

store.subscribe(() =>
  console.log(store.getState())
)


store.dispatch({ type: 'INCREMENT' })
// 1
store.dispatch({ type: 'INCREMENT' })
// 2
store.dispatch({ type: 'DECREMENT' })
// 1
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 212,686评论 6 492
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,668评论 3 385
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 158,160评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,736评论 1 284
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 65,847评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,043评论 1 291
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,129评论 3 410
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,872评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,318评论 1 303
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,645评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,777评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,470评论 4 333
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,126评论 3 317
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,861评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,095评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,589评论 2 362
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,687评论 2 351

推荐阅读更多精彩内容