Redux详细入门教程(从0到1)

React

2023年02月09日 11:24:094025

你真的需要Redux吗?

Redux 可帮助你处理共享状态的管理,但与任何工具一样,它也需要权衡利弊。使用 Redux 有更多的概念需要学习,还有更多的代码需要编写,需要添加了一些额外代码,并要求你遵循某些限制。这是短期和长期生产力之间的权衡。

在以下情况下使用 Redux:

  • 应用中有很多 state 在多个组件中需要使用

  • 应用 state 会随着时间的推移而频繁更新

  • 更新 state 的逻辑很复杂

  • 中型和大型代码量的应用,很多人协同开发

如果你的项目数据来源单一,没有过多的状态传递,也没有使用WebSocket等需要共享状态的功能,那么,Redux是不必须的。

Redux是什么

Redux 是一个使用叫做 “action” 的事件来管理和更新应用状态的模式和工具库 它以集中式 Store(centralized store)的方式对整个应用中使用的状态进行集中管理,其规则确保状态只能以可预测的方式更新。

//cdn.shiniest.cn/static/202302/redux_data_flow.png

redux数据流

Redux架构原理:

  1. 剥离组件数据(state)

  2. 数据统一存放在store中

  3. 组件订阅store获得数据

  4. store推送数据更新

类似与vue的状态管理vuex

一句话总结:Redux统一保存了数据,在隔离了数据与UI的同时,负责处理数据的绑定。

使用Redux

$ yarn add redux
# or
$ npm install redux --save

原则

  1. store是唯一的 (整个应用只能有一个 store)

  2. 只有store能改变自己的内容 (store里的数据不是reducer更新的)

  3. reducer必须是纯函数 (只要是同样的输入,必定得到同样的输出)

核心API

//cdn.shiniest.cn/static/202302/redux_api.png

1. Store

Store 就是保存全局数据的地方,你可以把它看成一个容器(带有推送功能的数据仓库)。整个应用只能有一个 Store

Redux 提供createStore这个函数,用来生成 Store。

import { createStore } from 'redux';
const store = createStore(fn);

2. State

Store对象包含所有数据。如果想得到某个时点的数据,就要对 Store 生成快照。这种时点的数据集合,就叫做 State。

当前时刻的 State,可以通过store.getState()拿到。

import { createStore } from 'redux';
const store = createStore(fn);

const state = store.getState();

Redux 规定, 一个 State 对应一个 View。只要 State 相同,View 就相同。你知道 State,就知道 View 是什么样,反之亦然。

3. Action

State 的变化,会导致 View 的变化。但是,用户接触不到 State,只能接触到 View。所以,State 的变化必须是 View 导致的。

Action 就是 View 发出的通知,表示 State 应该要发生变化了。

Action 是一个对象。其中的type属性是必须的,表示 Action 的名称。其他属性可以自由设置。

const action = {
  type: 'indexChange',
  value: 1,
};

上面代码中,Action 的名称是indexChange,它携带的参数是boolean类型的 1

可以这样理解,Action 描述当前发生的事情。改变 State 的唯一办法,就是使用 Action。它会运送数据到 Store。

4. Action Creator

过多的Action在组件中,会显得代码过于臃肿,不利于阅读和维护
可以新建一个actionCreator.js文件,用来统一管理action

// action的统一管理

export const changeInputAction = value => ({
    type: 'change_input',
    value,
});

export const addItemAction = () => ({
    type: 'add_item',
});

export const delItemAction = index => ({
    type: 'del_item',
    index,
});

export const initListAction = list => ({
    type: 'init_list',
    list,
});

每个函数都用来返回一个action,这个函数就叫 Action Creator

5. store.dispatch()

store.dispatch()是 View 发出 Action 的唯一方法

import { createStore } from 'redux';
const store = createStore(fn);

const action = {
    type: 'change_input',
    value: 'Learn Redux',
};

store.dispatch(action);

结合 Action Creator,这段代码可以改写如下:

import { createStore } from 'redux';
import { changeInputAction } from 'src/store/actionCreator.js'

const store = createStore(fn);

store.dispatch(changeInputAction('Learn Redux'));

6. Reducer

Store 收到 Action 以后,必须给出一个新的 State,这样 View 才会发生变化。这种 State 的计算过程就叫做 Reducer。

Reducer 是一个函数,它接受 Action 和当前 State 作为参数,返回一个新的 State。

可以新建一个reducer.js文件,用来统一管理

src/store/reducer.js

// 初始状态,作为 State 的默认值
const defaultState = {
    value: ''
};

const reducer = (state = defaultState, action) => {
    let newState = JSON.parse(JSON.stringify(state)); // 深拷贝,不能直接修改state里的数据
    if (action.type === 'change_input') {
        newState.value = action.value;
    }
    if (action.type === 'add_item') {
        newState.list.push(newState.value);
        newState.value = '';
    }
    if (action.type === 'del_item') {
        newState.list.splice(action.index, 1);
    }
    if (action.type === 'init_list') {
        newState.list = action.list;
    }
    return newState;
};

export default reducer;

reducer函数收到名为change_inputadd_itemdel_iteminit_list的 Action 后,就返回一个新的 State(newState),作为结果抛出。

在生成 Store 的时候,将 reducer传入createStore()。以后每当store.dispatch发送过来一个新的 Action,就会自动调用 reducer,得到新的 State

import { createStore } from 'redux';
import reducer from 'src/store/reducer.js';

const store = createStore(reducer);

7. store.subscribe()

Store 允许使用store.subscribe()方法设置监听函数一旦 State 发生变化,就自动执行这个函数

import { Component} from 'react';
import store from './store/index';

class List extends Component {
  constructor(props) {
    super(props);
    store.subscribe(this.storeChange.bind(this)); // store发生改变时,自动触发storeChange方法
  }
  render() {
    return //...
  }
  storeChange() {
    // store发生改变时,将自动触发
    let newState = store.getState();  // 得到当前state状态,包含所有store属性
    this.setState(newState);  // 重新渲染 View
  }
}

export default List;

store.subscribe方法返回一个函数,调用这个函数就可以解除监听

let unsubscribe = store.subscribe(() =>
  console.log(store.getState())
);

unsubscribe(); // 解除监听

工作流程

//cdn.shiniest.cn/static/202302/ReduxDataFlowDiagram.gif

1. 用户发出 Action

const action = {
  type: 'change_input',
  value: 'Learn Redux',
};

store.dispatch(action);

2. Store 自动调用 Reducer,并且传入两个参数:当前 State 和收到的 Action。 Reducer 会返回新的 State(newState)

const reducer = (state, action) => {
  let newState = JSON.parse(JSON.stringify(state)); // 深拷贝,不能直接修改state里的数据
  if (action.type === 'change_input') {
      newState.value = action.value;
  }
  return newState;
};

export default reducer;

3. State 一旦有变化,Store 就会调用监听函数

store.subscribe(this.storeChange.bind(this)); // store发生改变时,自动触发storeChange方法

4.storeChange()可以通过store.getState()得到当前状态。如果使用的是 React,这时可以触发重新渲染 View

storeChange() {
  // store发生改变时,将自动触发
  let newState = store.getState();  // 得到当前state状态,包含所有store属性
  this.setState(newState);  // 重新渲染 View
}

完整示例

src/store/index.js

import { createStore } from 'redux';
import reducer from './reducer';

const store = createStore(reducer, window.__REDUX_DEVTOOLS_EXTENSION_COMPOSE__ && window.__REDUX_DEVTOOLS_EXTENSION_COMPOSE__());

export default store;

src/list.js

import store from './store/index';
import { changeInputAction } from './store/actionCreator';

class List extends Component {
  constructor(props) {
    super(props);
    this.state = store.getState();  // 获取store中所有的数据
    store.subscribe(this.storeChange.bind(this)); // store发生改变时,自动触发
  }
  render() {
    return (
      <div>
          <input value={this.state.value} onChange={this.change.bind(this)} />
      </div>
    );
  }
  storeChange() {
    this.setState(store.getState());
  }
  change(e) {
    const action = changeInputAction(e.target.value);
    store.dispatch(action); // 派发action给store
  }
}

export default List;

src/store/actionCreator.js

export const changeInputAction = value => ({
  type: 'change_input',
  value,
});

src/store/reducer.js

const defaultState = {
  value: ''
};

const reducer = (state = defaultState, action) => {
  console.log(state, action);
  let newState = JSON.parse(JSON.stringify(state)); // 深拷贝,不能直接修改state里的数据
  if (action.type === 'change_input') {
    newState.value = action.value;
  }
  return newState;
};

export default reducer;
赞 5
收藏
分享

本作品系 原创,作者:你不熟悉的x先生

原文链接:https://shiniest.cn/blog/article/159

文本版权:文本版权归作者所有

转载需著名并注明出处(禁止商业使用)

评论和回复

0/500
    没有更多啦~
    怎么一条数据都没有呢?