react-redux流程与实现分析

1、前言

redux作为一种单向数据流的实现,配合react非常好用,尤其是在项目比较大,逻辑比较复杂的时候,单项数据流的思想能使数据的流向、变化都能得到清晰的控制,并且能很好的划分业务逻辑和视图逻辑。本文主要记录下自己学习redux、react-redux时的理解。



上图展示了redux数据的基本流程,简单的说就是view dispatch一个action后,通过对应reducer处理,然后更新store,最终views根据store数据的改变重新渲染界面。

2、创建store

store就是redux的一个数据中心,简单的理解就是我们所有的数据都会存放在里面,然后在界面上使用时,从中取出对应的数据。因此最开始,我们要创建一个这样的store,redux提供了createStore方法。

export default function createStore(reducer, preloadedState, enhancer) {
  ...
  return {
    dispatch,
    subscribe,
    getState,
    replaceReducer,
    [$$observable]: observable
  }
}

可以看到createStore有三个参数,返回一个对象,里面有我们常用的方法,下面一一来看一下。

(1)、getState

export default function createStore(reducer, preloadedState, enhancer) {
  ...
  var currentState = preloadedState
  function getState() {
    return currentState
  }
  ...
}

代码只有一行,redux内部通过currentState变量保存当前store,变量初始值即我们调用时传进来的preloadedState,getState()就是返回这个变量

(2)、subscribe
  代码本身也不难,就是通过nextListeners数组保存所有的回调函数,外部调用subscribe时,会将传入的listener插入到nextListeners数组中,并返回unsubscribe函数,通过此函数可以删除nextListeners中对应的回调。这里有个小细节是使用了currentListeners和nextListeners两个数组来保存,主要原因是在dispatch函数中会遍历nextListeners,这时候可能会客户可能会继续调用subscribe插入listener,为了保证遍历时nextListeners不变化,需要一个临时的数组保存。

var currentListeners = []
var nextListeners = currentListeners

function ensureCanMutateNextListeners() {
    if (nextListeners === currentListeners) {
      nextListeners = currentListeners.slice()        //生成一个新的数组
    }
 }

function subscribe(listener) {
    if (typeof listener !== 'function') {
      throw new Error('Expected listener to be a function.')
    }

    var isSubscribed = true

    ensureCanMutateNextListeners()
    nextListeners.push(listener)

    return function unsubscribe() {
      if (!isSubscribed) {
        return
      }

      isSubscribed = false

      ensureCanMutateNextListeners()
      var index = nextListeners.indexOf(listener)
      nextListeners.splice(index, 1)
    }
 }

(3)、dispatch
  我们知道dispatch一个action后,就会调用此action对应的reducer,从下面源码可以看的很清晰,在调用了currentReducer以后,遍历nextListeners数组,回调所有通过subscribe注册的函数。这样在每次store数据更新,组件就能立即获取到最新的数据

function dispatch(action) {  
  ...
  try {
      isDispatching = true
      currentState = currentReducer(currentState, action)  //调用reducer处理
    } finally {
      isDispatching = false
    }

    var listeners = currentListeners = nextListeners
    for (var i = 0; i < listeners.length; i++) {                   
      var listener = listeners[i]
      listener()
    }
  ...
}

(4)、replaceReducer
  replaceReducer是切换当前的reducer,虽然代码只有几行,但是在用到时功能非常强大,它能够实现代码热更新的功能,即在代码中根据不同的情况,对同一action调用不同的reducer,从而得到不同的数据。

3、中间件
  前面我们分析了createStore方法,通过createStore创建的store已经可以满足基本的分发action、修改store、更新view的功能了。但是仔细思考,对于前端必须用到的异步请求却无能为力,因为dispatch一个异步action时,数据还没有返回,而此时已经调用了currentReducer()方法,这样就没法更新store了。这里就需要用到中间件了。

  什么是中间件呢,我的理解是:比如水从自来水厂流到用户的家中,需要经过过滤、消毒、净化等一道道工艺,最终才会到用户的家中,这中间的一道道工艺就可以理解为中间件,每个中间件都可以对水进行一些处理,当然也可以不处理,直接让水流过去。而我们的数据类似于水,中间件就是对数据进行特殊处理的一个个节点。redux提供了applyMiddleware方法,在分析此方法之前,先看下redux中另一个方法,compose。

export default function compose(...funcs) {
  if (funcs.length === 0) {
    return arg => arg
  }

  funcs = funcs.filter(func => typeof func === 'function')

  if (funcs.length === 1) {
    return funcs[0]
  }

  const last = funcs[funcs.length - 1]
  const rest = funcs.slice(0, -1)
  return (...args) => rest.reduceRight((composed, f) => f(composed), last(...args))
}

这里有一个数学概念需要知道,叫柯里化函数,什么是柯里化函数,看一个简单的例子

var add = function (a) {
  return function (b) {
    return a + b;
  }
}

var two = add(2)
var three = two(1)
var four = two(2)            //与 add(2)(2)效果一样

如果我们定义一般的函数实现加法,那么需要传入两个参数,而add方法每次只传入一个参数,即把接受多个参数的函数变换成接受一个单一参数(最初函数的第一个参数)的函数。
  再回过头来看compose就会比较清晰,它类似与上面的two函数,后续每次调用,都与之前传入的2相加。而compose返回的函数,在后续继续调用时,就会遍历调用传入compose的funcs数组,compose就是对传入的funcs数组进行柯里化。理解了这个方法后,再来看applyMiddleware。applyMiddleware的参数就是一系列中间件。通过compose调用每个中间件,并传入原生dispatch方法,然后返回增强后的dispatch方法,最终返回store。由此可见applyMiddleware方法其实就是增强dispatch,这样在使用dispatch时可以让action流过中间件。

export default function applyMiddleware(...middlewares) {
  return (createStore) => (reducer, preloadedState, enhancer) => {
    var store = createStore(reducer, preloadedState, enhancer)
    var dispatch = store.dispatch
    var chain = []

    var middlewareAPI = {
      getState: store.getState,
      dispatch: (action) => dispatch(action)
    }
    chain = middlewares.map(middleware => middleware(middlewareAPI))
    dispatch = compose(...chain)(store.dispatch)

    return {
      ...store,
      dispatch
    }
  }
}

下面来看一下中间件的写法

const funActionThunk = ({ dispatch, getState }) => next => action => {
    if (typeof action.payload === 'function') {
        return action.payload(dispatch, getState);
    }
    return next(action);
};
export default funActionThunk;

我们可以来一一看一下中间件的这些参数是从哪里传进来的,首先{ dispatch, getState }是在applyMiddleware中传入的middlewareAPI对象,也就是原始的dispatch和getState对象。next参数其实也是原始的dispatch,就是compose(...chain)(store.dispatch)调用时传入的store.dispatch。action就是我们增强的dispatch分发的action。参数都传入后,我们就可以依据不同的action进行处理了,比如上面的中间件对action.payload进行判断,如果是函数,就直接调用,如果不是,那么调用next,直至最后没有中间件了,此时next就是原始的dispatch(即最初传入的store.dispatch)。

4、bindActionCreators

bindActionCreators方法的目的就是简化action的分发,我们在触发一个action时,最基本的调用是dispatch(action(param))。这样需要在每个调用的地方都写dispatch,非常麻烦。bindActionCreators就是将action封装了一层,返回一个封装过的对象,此后我们要出发action时直接调用action(param)就可以了。

5、react-redux

redux其实是一个通用的库,它不只针对react,还可以用到其它的像vue等库。因此react要想完美的应用redux,还需要封装一层,react-redux就是此作用。react-redux库相对简单些,它提供了一个react组件Provider和一个方法connect。下面的代码是react-redux项目的一般写法。

<Provider store={store} >
    <div style={{height: '100%'}}>
        <Handler/>
    </div>
</Provider>

Provider组件相当于一个外壳,将store传入,并提供getChildContext方法,让后续的子组件可以通过context取到store对象。

connect方法复杂点,它返回一个函数,此函数的功能是创建一个connect组件包在WrappedComponent组件外面,connect组件复制了WrappedComponent组件的所有属性,并通过redux的subscribe方法注册监听,当store数据变化后,connect就会更新state,然后通过mapStateToProps方法选取需要的state,如果此部分state更新了,connect的render方法就会返回新的组件。

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

推荐阅读更多精彩内容