一文总结redux、react-redux、redux-thunk、redux-sage | 掘金技术征文-双节特别篇

7,430 阅读19分钟

redux系列总结

前言

hello大家好,我是风不识途,最近一直在整理redux系列文章,redux对初学者不太友好,关系错综复杂,难倒是不太难,就是比较复杂,所以这篇文章带你全面总结(根据王红元老师教学总结)redux、react-redux、redux-thunk还有redux-sage,immutable知识点(多图预警),由于知识点比较多,建议先收藏(收藏等于学会了),对你有用的话就给个赞👍


认识纯函数

JavaScript纯函数

  • 函数式编程中有一个概念叫纯函数, JavaScript符合函数式编程的范式, 所以也有纯函数的概念

  • React中,纯函数的概念非常重要,在接下来我们学习的Redux中也非常重要,所以我们必须来回顾一下纯函数

  • 纯函数的维基百科定义(了解即可) 在程序设计中,若一个函数符合一下条件,那么这个函数被称为纯函数:
    此函数在相同的输入值时, 需产生相同的输出
    函数的输出和输入值以外的其他隐藏信息或状态无关, 也由I/O设备产生的
    该函数不能有语义上可观察的函数副作用,诸如“触发事件”,使输出设备输出,或更改输出值以外物件的内容等
  • 纯函数的定义简单总结一下:

    • 纯函数指的是, 每次给相同的参数, 一定返回相同的结果
    • 函数在执行过程中, 不能产生副作用
  • 纯函数( Pure Function )的注意事项:
    在纯函数中不能使用随机数
    不能使用当前的时间或日期, 因为结果是会变的
    不能使用或者修改全局状态, 比如DOM,文件、数据库等等(因为如果全局状态改变了,它就会影响函数的结果)
    纯函数中的参数不能变化,否则函数的结果就会改变

React中的纯函数

  • 为什么纯函数在函数式编程中非常重要呢?
    • 因为你可以安心的写和安心的用
    • 你在写的时候保证了函数的纯度,实现自己的业务逻辑即可,不需要关心传入的内容或者函数体依赖了外部的变量
    • 你在用的时候,你确定你的输入内容不会被任意篡改,并且自己确定的输入,一定会有确定的输出
  • React非常灵活,但它也有一个严格的规则:
    • 所有React组件都必须像"纯函数"一样保护它们的"props"不被更改

认识Redux

为什么需要redux

  • JavaScript开发的应用程序, 已经变得非常复杂了:

    • JavaScript需要管理的状态越来越多, 越来越复杂了
    • 这些状态包括服务器返回的数据, 用户操作的数据等等, 也包括一些UI的状态
  • 管理不断变化的state是非常困难的:

    • 状态之间相互存在依赖, 一个状态的变化会引起另一个状态的变化, View页面也有可能会引起状态的变化
    • 当程序复杂时, state在什么时候, 因为什么原因发生了变化, 发生了怎样的变化, 会变得非常难以控制和追踪

React的作用

  • React只是在视图层帮助我们解决了DOM的渲染过程, 但是state依然是留给我们自己来管理:

    • 无论是组件定义自己的state,还是组件之间的通信通过props进行传递
    • 也包括通过Context进行数据之间的共享
    • React主要负责帮助我们管理视图,state如何维护最终还是我们自己来决定
  • Redux就是一个帮助我们管理State的容器:

    • ReduxJavaScript的状态容器, 提供了可预测的状态管理
  • Redux除了和React一起使用之外, 它也可以和其他界面库一起来使用(比如Vue), 并且它非常小 (包括依赖在内,只有2kb)

Redux的核心理念-Store

  • Redux的核心理念非常简单

  • 比如我们有一个朋友列表需要管理:

    • 如果我们没有定义统一的规范来操作这段数据,那么整个数据的变化就是无法跟踪的
    • 比如页面的某处通过products.push的方式增加了一条数据
    • 比如另一个页面通过products[0].age = 25修改了一条数据
  • 整个应用程序错综复杂,当出现bug时,很难跟踪到底哪里发生的变化

Redux的核心理念-action

  • Redux要求我们通过action来更新state
    • 所有数据的变化, 必须通过dispatch来派发action来更新
    • action是一个普通的JavaScript对象,用来描述这次更新的typecontent
  • 比如下面就是几个更新friendsaction:
    • 强制使用action的好处是可以清晰的知道数据到底发生了什么样的变化,所有的数据变化都是可跟追踪、可预测的
    • 当然,目前我们的action是固定的对象,真实应用中,我们会通过函数来定义,返回一个action

Redux的核心理念-reducer

  • 但是如何将stateaction联系在一起呢? 答案就是reducer
    • reducer是一个纯函数
    • reducer做的事情就是将传入的stateaction结合起来来生成一个新的state

Redux的三大原则

  • 单一数据源
    • 整个应用程序的state被存储在一颗object tree中, 并且这个object tree只存储在一个store
    • Redux并没有强制让我们不能创建多个Store,但是那样做并不利于数据的维护
    • 单一的数据源可以让整个应用程序的state变得方便维护、追踪、修改
  • State是只读的
    • 唯一修改state的方法一定是触发action, 不要试图在其它的地方通过任何的方式来修改state
    • 这样就确保了View或网络请求都不能直接修改state,它们只能通过action来描述自己想要如何修改state
    • 这样可以保证所有的修改都被集中化处理,并且按照严格的顺序来执行,所以不需要担心race condition(竟态)的问题
  • 使用纯函数来执行修改
    • 通过reducer将旧 stateaction 联系在一起, 并且返回一个新的state
    • 随着应用程序的复杂度增加,我们可以将reducer拆分成多个小的reducers,分别操作不同state tree的一部分
    • 但是所有的reducer都应该是纯函数,不能产生任何的副作用

Redux的基本使用

Redux中核心的API

redux的安装: yarn add redux

  1. createStore 可以用来创建 store对象
  2. store.dispatch 用来派发 action , action 会传递给 store
  3. reducer接收action,reducer计算出新的状态并返回它 (store负责调用reducer)
  4. store.getState 这个方法可以帮助获取 store 里边所有的数据内容
  5. store.subscribe方法可以让让我们订阅 store 的改变,只要 store 发生改变, store.subscribe 这个函数接收的这个回调函数就会被执行

小结

  1. 创建sotore, 决定 store 要保存什么状态
  2. 创建action, 用户在程序中实现什么操作
  3. 创建reducer, reducer 接收 action 并返回更新的状态

Redux的使用过程

  1. 创建一个对象, 作为我们要保存的状态
  2. 创建Store来存储这个state
    • 创建store时必须创建reducer
    • 我们可以通过 store.getState 来获取当前的state
  3. 通过action来修改state
    • 通过dispatch来派发action
    • 通常action中都会有type属性,也可以携带其他的数据
  4. 修改reducer中的处理代码
    • 这里一定要记住,reducer是一个纯函数,不能直接修改state
    • 后面会讲到直接修改state带来的问题
  5. 可以在派发action之前,监听store的变化
import { createStore } from 'redux'

// 1.初始化state
const initState = { counter: 0 }

// 2.reducer纯函数 不能修改传递的state
function reducer(state = initState, action) {
  switch (action.type) {
    case 'INCREMENT':
      return { ...state, counter: state.counter + 1 }
    case 'ADD_COUNTER':
      return { ...state, counter: state.counter + action.num }
    default:
      return state
  }
}

// 3.store 参数放一个reducer
const store = createStore(reducer)

// 4.action
const action1 = { type: 'INCREMENT' }
const action2 = { type: 'ADD_COUNTER', num: 2 }

// 5.订阅store的修改
store.subscribe(() => {
  console.log('state发生了改变: ', store.getState().counter)
})

// 6.派发action
store.dispatch(action1)
store.dispatch(action2)

redux原理图

Redux结构划分

  • 如果我们将所有的逻辑代码写到一起, 那么当redux变得复杂时代码就难以维护
  • 对代码进行拆分, 将store、reducer、action、constants拆分成一个个文件
拆分目录

Redux使用流程

redux

Redux官方流程图

redux-flow


React-Redux的使用

redux融入react代码(案例)

  • redux融入react代码案例:

    • Home组件:其中会展示当前的counter值,并且有一个+1和+5的按钮
    • Profile组件:其中会展示当前的counter值,并且有一个-1和-5的按钮

  • 核心代码主要是两个:

    • componentDidMount 中订阅数据的变化,当数据发生变化时重新设置 counter
    • 在发生点击事件时,调用storedispatch来派发对应的action

自定义connect函数

当我们多个组件使用redux时, 重复的代码太多了, 比如: 订阅state取消订阅state 或 派发action获取state

将重复的代码进行封装, 将不同的statedispatch作为参数进行传递

//  connect.js 
import React, { PureComponent } from 'react'
import { StoreContext } from './context'
/**
 * 1.调用该函数: 返回一个高阶组件
 *      传递需要依赖 state 和 dispatch 来使用state或通过dispatch来改变state
 *
 * 2.调用高阶组件:
 *      传递该组件需要依赖 store 的组件
 *
 * 3.主要作用:
 *      将重复的代码抽取到高阶组件中,并将该组件依赖的 state 和 dispatch
 *      通过调用mapStateToProps()或mapDispatchToProps()函数
 *      并将该组件依赖的state和dispatch供该组件使用,其他使用store的组件不必依赖store
 *
 * 4.connect.js: 优化依赖
 *      目的:但是上面的connect函数有一个很大的缺陷:依赖导入的 store
 *      优化:正确的做法是我们提供一个Provider,Provider来自于我们
 * 			Context,让用户将store传入到value中即可;
 */
export function connect(mapStateToProps, mapDispatchToProps) {
  return function enhanceComponent(WrapperComponent) {
    class EnhanceComponent extends PureComponent {
      constructor(props, context) {
        super(props, context)

        // 组件依赖的state
        this.state = {
          storeState: mapStateToProps(context.getState()),
        }
      }

      // 订阅数据发生变化,调用setState重新render
      componentDidMount() {
        this.unsubscribe = this.context.subscribe(() => {
          this.setState({
            centerStore: mapStateToProps(this.context.getState()),
          })
        })
      }

      // 组件被卸载取消订阅
      componentWillUnmount() {
        this.unsubscribe()
      }

      render() {
        // 下面的WrapperComponent相当于 home 组件(就是你传递的组件)
        // 你需要将该组件需要依赖的state和dispatch作为props进行传递
        return (
          <WrapperComponent
            {...this.props}
            {...mapStateToProps(this.context.getState())}
            {...mapDispatchToProps(this.context.dispatch)}
          />
        )
      }
    }
    // 取出Provider提供的value
    EnhanceComponent.contextType = StoreContext
    return EnhanceComponent
  }
}

// home.js
// 定义组件依赖的state和dispatch
const mapStateToProps = state => ({
  counter: state.counter,
})

const mapDispatchToProps = dispatch => ({
  increment() {
    dispatch(increment())
  },
  addNumber(num) {
    dispatch(addAction(num))
  },
})
export default connect(mapStateToProps,mapDispatchToProps)(依赖redux的组件)

react-redux使用

  • 开始之前需要强调一下,reduxreact没有直接的关系,你完全可以在React, Angular, Ember, jQuery, or vanilla JavaScript中使用Redux

  • 尽管这样说,redux依然是和React或者Deku的库结合的更好,因为他们是通过state函数来描述界面的状态,Redux可以发射状态的更新,让他们作出相应。

  • 虽然我们之前已经实现了connectProvider这些帮助我们完成连接redux、react的辅助工具,但是实际上redux官方帮助我们提供了 react-redux 的库,可以直接在项目中使用,并且实现的逻辑会更加的严谨和高效

  • 安装react-redux

    • yarn add react-redux
// 1.index.js
import { Provider } from 'react-redux'
ReactDOM.render(
  <Provider store={store}>
    <App />
  </Provider>,
  document.getElementById('root')
)

// 2.home.js
import { connect } from 'react-redux'
// 定义需要依赖的state和dispatch (函数需要返回一个对象)
export default connect(mapStateToProps, mapDispatchToProps)(About)

react-redux源码导读


Redux-Middleware中间件

组件中异步操作

  • 在之前简单的案例中,redux中保存的counter是一个本地定义的数据

    • 我们可以直接通过同步的操作来dispatch actionstate就会被立即更新。

    • 但是真实开发中,redux中保存的很多数据可能来自服务器,我们需要进行异步的请求,再将数据保存到redux

  • 网络请求可以在class组件的componentDidMount中发送,所以我们可以有这样的结构:

redux中异步操作

  • 上面的代码有一个缺陷:

    • 我们必须将网络请求的异步代码放到组件的生命周期中来完成
  • 为什么将网络请求的异步代码放在redux中进行管理?

    • 后期代码量的增加,如果把网络请求异步函数放在组件的生命周期里,这个生命周期函数会变得越来越复杂,组件就会变得越来越大
    • 事实上,网络请求到的数据也属于状态管理的一部分,更好的一种方式应该是将其也交给redux来管理

  • 但是在redux中如何可以进行异步的操作呢?
    • 使用中间件 (Middleware)
    • 学习过ExpressKoa框架的童鞋对中间件的概念一定不陌生
    • 在这类框架中,Middleware可以帮助我们在请求和响应之间嵌入一些操作的代码,比如cookie解析、日志记录、文件压缩等操作

理解中间件(重点)

  • redux也引入了中间件 (Middleware) 的概念:
    • 这个中间件的目的是在dispatchaction和最终达到的reducer之间,扩展一些自己的代码
    • 比如日志记录、调用异步接口、添加代码调试功能等等

redux-middlware

  • redux-thunk是如何做到让我们可以发送异步的请求呢?
    • 默认情况下的dispatch(action)action需要是一个JavaScript的对象
    • redux-thunk可以让dispatch(action函数), action可以是一个函数
    • 该函数会被调用, 并且会传给这个函数两个参数: 一个dispatch函数和getState函数
      • dispatch函数用于我们之后再次派发action
      • getState函数考虑到我们之后的一些操作需要依赖原来的状态,用于让我们可以获取之前的一些状态

redux-thunk的使用

  1. 安装redux-thunk

    • yarn add redux-thunk
  2. 在创建store时传入应用了middlewareenhance函数

    • 通过applyMiddleware来结合多个Middleware, 返回一个enhancer

    • enhancer作为第二个参数传入到createStore

      image-20200821182447344

  3. 定义返回一个函数的action

    • 注意:这里不是返回一个对象了,而是一个函数
    • 该函数在dispatch之后会被执行

查看代码
import { createStore, applyMiddleware } from 'redux'
import reducer from './reducer'
import thunk from 'redux-thunk'

const store = createStore(
  reducer,
  applyMiddleware(thunk) // applyMiddleware可以使用中间件模块
) 
export default store

redux-devtools

redux-devtools插件

  • 我们之前讲过,redux可以方便的让我们对状态进行跟踪和调试,那么如何做到呢?
    • redux官网为我们提供了redux-devtools的工具
    • 利用这个工具,我们可以知道每次状态是如何被修改的,修改前后的状态变化等等
  • 使用步骤:
    • 第一步:在浏览器上安装redux-devtools扩展插件
    • 第二步:在redux中集成devtools的中间件
// store.js 开启redux-devtools扩展
import { createStore, applyMiddleware, compose } from 'redux'

// composeEnhancers函数
const composeEnhancers =
  window.__REDUX_DEVTOOLS_EXTENSION_COMPOSE__({ trace: true }) || compose

// 通过applyMiddleware来结合多个Middleware,返回一个enhancer
const enhancer = applyMiddleware(thankMiddleware)

// 通过enhancer作为第二个参数传递createStore中
const store = createStore(reducer, composeEnhancers(enhancer))

export default store

redux-sage

generator

Generator 函数是 ES6 提供的一种异步编程解决方案,语法行为与传统函数完全不同

Generator 函数有多种理解角度。语法上,首先可以把它理解成,Generator 函数是一个状态机,封装了多个内部状态。

// 生成器函数的定义
// 默认返回: Generator
function* foo() {
  console.log('111')
  yield 'hello'
  console.log('222')
  yield 'world'
  console.log('333')
  yield 'jane'
  console.log('444')
}
// iterator: 迭代器
const result = foo()
console.log(result)

// 使用迭代器
// 调用next,就会消耗一次迭代器
const res1 = result.next()
console.log(res1) // {value: "hello", done: false}
const res2 = result.next()
console.log(res2) // {value: "world", done: false}
const res3 = result.next()
console.log(res3) // {value: "jane", done: false}
const res4 = result.next()
console.log(res4) // {value: undefined, done: true}

redux-sage流程

redux-saga的使用

  • redux-saga是另一个比较常用在redux发送异步请求的中间件,它的使用更加的灵活

  • Redux-saga的使用步骤如下

    1. 安装redux-sage: yarn add redux-saga

    2. 集成redux-saga中间件

      • 引入 createSagaMiddleware 后, 需要创建一个 sagaMiddleware

      • 然后通过 applyMiddleware 使用这个中间件,接着创建 saga.js 这个文件

      • 启动中间件的监听过程, 并且传入要监听的saga

    3. saga.js文件的编写

      • takeEvery:可以传入多个监听的actionType,每一个都可以被执行(对应有一个takeLatest,会取消前面的)
      • put:在saga中派发action不再是通过dispatch, 而是通过put
      • all:可以在yield的时候put多个action
// store.js
import createSageMiddleware from 'redux-saga'
import saga from './saga'
// 1.创建sageMiddleware中间件
const sagaMiddleware = createSageMiddleware()
// 2.应用一些中间件
const enhancer = applyMiddleware(sagaMiddleware)
const store = createStore(reducer,composeEnhancers(enhancer))

sagaMiddleware.run(saga)
export default store

// saga.js
import { takeEvery, put, all } from 'redux-saga/effects'
import { FETCH_HOME_DATA } from './constant'

function* fetchHomeData(action) {
  const res = yield axios.get('http://123.207.32.32:8000/home/multidata')
  const banners = res.data.data.banner.list
  const recommends = res.data.data.recommend.list
  // dispatch action 提交action,redux-sage提供了put
  yield all([
    yield put(changeBannersAction(banners)),
    yield put(changeRecommendAction(recommends)),
  ])
}

function* mySaga() {
  // 参数一:要拦截的actionType
  // 参数二:生成器函数
  yield all([
    takeEvery(FETCH_HOME_DATA, fetchHomeData),
  ])
}

export default mySaga

reducer代码拆分

Reducer代码拆分

  • 我们来看一下目前我们的reducer

    • 当前这个reducer既有处理counter的代码,又有处理home页面的数据

    • 后续counter相关的状态或home相关的状态会进一步变得更加复杂

    • 我们也会继续添加其他的相关状态,比如购物车、分类、歌单等等

    • 如果将所有的状态都放到一个reducer中进行管理,随着项目的日趋庞大,必然会造成代码臃肿、难以维护

  • 因此,我们可以对reducer进行拆分:

    • 我们先抽取一个对counter处理的reducer
    • 再抽取一个对home处理的reducer
    • 将它们合并起来

Reducer文件拆分

  • 目前我们已经将不同的状态处理拆分到不同的reducer中,我们来思考:

    • 虽然已经放到不同的函数了,但是这些函数的处理依然是在同一个文件中,代码非常的混乱

    • 另外关于reducer中用到的constantaction等我们也依然是在同一个文件中;

combineReducers函数

  • 目前我们合并的方式是通过每次调用reducer函数自己来返回一个新的对象
  • 事实上,redux给我们提供了一个combineReducers函数可以方便的让我们对多个reducer进行合并
import { combineReducers } from 'redux'
import { reducer as counterReducer } from './count'
import { reducer as homeReducer } from './home'

export const reducer = combineReducers({
  counterInfo: counterReducer,
  homeInfo: homeReducer,
})
  • 那么combineReducers是如何实现的呢?
    • 它将我们传递的reducer合并成一个对象, 最终返回一个combination函数
    • 在执行combination函数过程中, 会通过判断前后返回的数据是否相同来决定返回之前的state还是新的state

immutableJs

数据可变形的问题

  • React开发中,我们总是会强调数据的不可变性:

    • 无论是类组件中的state,还是reduex中管理的state
    • 事实上在整个JavaScript编码的过程中,数据的不可变性都是非常重要的
  • 数据的可变性引发的问题(案例):

    • 我们明明没有修改obj,只是修改了obj2,但是最终obj也被我们修改掉了

    • 原因非常简单,对象是引用类型,它们指向同一块内存空间,两个引用都可以任意修改

const obj1 = { name: 'jane', age: 18 }
const obj2 = obj1
obj1.name = 'kobe'
console.log(obj2.name) // kobe
  • 有没有办法解决上面的问题呢?

    • 进行对象的拷贝即可:Object.assign或扩展运算符
  • 这种对象的浅拷贝有没有问题呢?

    • 从代码的角度来说,没有问题,也解决了我们实际开发中一些潜在风险

    • 从性能的角度来说,有问题,如果对象过于庞大,这种拷贝的方式会带来性能问题以及内存浪费

  • 有人会说,开发中不都是这样做的吗?

    • 从来如此,便是对的吗?

认识ImmutableJS

  • 为了解决上面的问题,出现了Immutable对象的概念:
    • Immutable对象的特点是只要修改了对象,就会返回一个新的对象,旧的对象不会发生改变;
  • 但是这样的方式就不会浪费内存了吗?
    • 为了节约内存,又出现了一个新的算法:Persistent Data Structure(持久化数据结构或一致性数据结构)
  • 当然,我们一听到持久化第一反应应该是数据被保存到本地或者数据库,但是这里并不是这个含义:
    • 用一种数据结构来保存数据
    • 当数据被修改时,会返回一个对象,但是新的对象会尽可能的利用之前的数据结构而不会对内存造成浪费,如何做到这一点呢?结构共享:

  • 安装Immutable: yarn add immutable

ImmutableJS常见API

注意:我这里只是演示了一些API,更多的方式可以参考官网

作用:不会修改原有数据结构,返回一个修改后新的拷贝对象

  • JavaScripImutableJS直接的转换
    • 对象转换成Immutable对象:Map
    • 数组转换成Immtable数组:List
    • 深层转换:fromJS
const im = Immutable
// 对象转换成Immutable对象
const info = {name: 'kobe', age: 18}
const infoIM = im.Map()

// 数组转换成Immtable数组
const names = ["abc", "cba", "nba"]
const namesIM = im.List(names)
  • ImmutableJS的基本操作:
    • 修改数据:set(property, newVal)
      • 返回值: 修改后新的数据结构
    • 获取数据:get(property/index)
    • 获取深层Immutable对象数据(子属性也是Immutable对象): getIn(['recommend', 'topBanners'])
// set方法 不会修改infoIM原有数据结构,返回修改后新的数据结构
const newInfo2IM = infoIM.set('name', 'james')
const newNamesIM = namesIM.set(0, 'why')

// get方法
console.log(infoIM.get('name'))// -> kobe
console.log(namesIM.get(0))// -> abc

结合Redux管理数据

  1. ImmutableJS重构redux

    • yarn add Immutable
    • yarn add redux-immutable
  2. 使用redux-immutable中的combineReducers;

  3. 所有的reducer中的数据都转换成Immutable类型的数据

FAQ

React中的state如何管理

  • 目前项目中采用的state管理方案(参考即可):

    • 相关的组件内部可以维护的状态,在组件内部自己来维护
    • 只要是需要共享的状态,都交给redux来管理和维护
    • 从服务器请求的数据(包括请求的操作) ,交给redux来维护
  • 🏆 掘金技术征文|双节特别篇