Redux原理分析

47次阅读

共计 7604 个字符,预计需要花费 20 分钟才能阅读完成。

Redux 原理分析

Redux 是什么

很多人认为 redux 必须要结合 React 使用,其实并不是的,Redux 是 JavaScript 状态容器,只要你的项目中使用到了状态,并且状态十分复杂,那么你就可以使用 Redux 管理你的项目状态,它可以使用在 react 中,也可以使用中在 Vue 中,当然也适用其他的框架。

一.redux 的工作原理

先上图(图片源于网络)

  1. 首先我们找到最上面的 state
  2. 在 react 中 state 决定了视图(ui),state 的变化就会调用 React 的 render() 方法,从而改变视图
  3. 用户通过一些事件(如点击按钮,移动鼠标)就会像 reducer 派发一个 action
  4. reducer 接收到 action 后就会去更新 state
  5. store 是包含了所有了 state,可以把他看做所有状态的集合

当然,现在可能看不懂这在瞎说啥,但是等把这篇文章看完再来这个图,和这段话,就会有恍然大明白的感觉

1.action

action 本质上就是一个对象,它一定有一个名为 type 的 key 如 {type: 'add'},{type: 'add'} 就是一个 action
但是我们只实际工作中并不是直接用 action,而是使用 action 创建函数,(千万别弄混淆),
顾名思义 action 创建函数就是一个函数,它的作用就是返回一个 action,如:

function add() {return {type: 'add'}
}

2.reducer

reducer 其实就是一个函数,它接收两个参数,第一个参数是需要管理的状态 state,第二个是 action。reducer 会根据传入的 action 的 type 值对 state 进行不同的操作,然后返回一个新的 state,而不是在原有 state 的基础上进行修改, 但是如果遇到了未知的(不匹配的)action,就会返回原有的 state,不进行任何改变

function reducer(state = {money: 0}, action) {// 返回一个新的 state 可以使用 es6 提供的 Object.assign()方法,或扩展运算符(此方法需要 babel-preset-state- 3 支持)switch (action.type) {
        case '+':
            return Object.assign({}, state, {money: state.money + 1});
        case '-':
            return {...state, ...{money: state.money - 1}};
        default:
            return state;
    }
}

3.store

你可以把 store 想成一个状态树,它包含了整个 redeux 应用的所有状态。
我们使用 redux 提供的 createStore 方法生成 store

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

store 提供了几个方法供我们使用,下面是我们常用的 3 个:

store.getState();// 获取整个状态树
store.dispatch();// 改变状态,改变 state 的唯一方法
store.subscribe();// 订阅一个函数,每当 state 改变时,都会去调用这个函数

接下来演示一个 redux 的完整应用,并且说明这三个方法该怎么用

import {createStore} from 'redux';

// 给初始状态一个默认值:{money: 0}
function reducer(state = {money: 0}, action) {// 返回一个新的 state 可以使用 es6 提供的 Object.assign()方法,或扩展运算符(此方法需要 babel-preset-state- 3 支持)switch (action.type) {
        case '+':
            return Object.assign({}, state, {money: state.money + 1});
        case '-':
            return {...state, ...{money: state.money - 1}};
        default:
            return state;
    }
}

//action 创建函数,返回了一个 action
function add() {return {type: '+'}
}

function subtraction() {return {type: '-'}
}

// 创建单一状态树
const store = createStore(reducer);

console.log(store.getState());//{money: 0}, 初始的状态,没有任何改变(通过 getState 来获取目前的状态)//store 通过 dispatch 这个方法,并且传入 action 作为参数,对 store 进行了改变
store.dispatch(add());
console.log(store.getState());//{money: 1},reducer 接受到了 '+' 这个命令,就捡到了一块钱

store.dispatch(subtraction());
console.log(store.getState());//{money: 0},reducer 接受到了 '-' 这个命令,又掉了一块钱

store.dispatch({type:'我是来捣乱的'});
console.log(store.getState());//{money: 0},reducer 接受到了一个不识别命令,返回原有的 state

这个时候我们就会发现几个问题:

  1. 每次状态改变的时候我们都要 console.log() 才能知道改变后的状态,
  2. action 的 type 实际上就是一个字符串,如果我们需要进行项目维护,更改 type 的值,就需要在多处进行修改,变得十分麻烦。

这个时候我们就可以使用 store.subscribe() 来订阅一个事件,代替我们在每次 dispatch 后都要 console.log() 后才能知道改变后的状态

function listen() {console.log(store.getState());
}

store.subscribe(listen);

将 type 维护成常量,这样我们在日后的维护过程中只需要对常量进行维护就可以了,我们目前这个 demo 使用到 type 的地方太少可能感觉不到,可是在实际项目中这个方法却非常的实用

const ADD = '+', SUBTRACTION = '-';

我们优化后的代码如下:

import {createStore} from 'redux';

// 定义常量方便维护
const ADD = '+', SUBTRACTION = '-';

// 给初始状态一个默认值:{money: 0}
function reducer(state = {money: 0}, action) {// 返回一个新的 state 可以使用 es6 提供的 Object.assign()方法,或扩展运算符(此方法需要 babel-preset-state- 3 支持)switch (action.type) {
        case ADD:
            return Object.assign({}, state, {money: state.money + 1});
        case SUBTRACTION:
            return {...state, ...{money: state.money - 1}};
        default:
            return state;
    }
}

//action 创建函数,返回了一个 action
function add() {return {type: ADD}
}

function subtraction() {return {type: SUBTRACTION}
}

// 打印改变后的状态
function listen() {console.log(store.getState());
}

// 创建单一状态树
const store = createStore(reducer);

// 订阅 listen,每次 dispatch 后都会执行 listen,从而打印状态(只有在执行 dispatch 后才会执行,状态初始化的时候并不会执行)store.subscribe(listen);

console.log(store.getState());// 初始的状态,没有任何改变

//store 通过 dispatch 这个方法,并且传入 action 作为参数,对 store 进行了改变
store.dispatch(add());
store.dispatch(subtraction());
store.dispatch({type: '我是来捣乱的'});

/* 控制台的打印结果如下:{money: 0}
{money: 1}
{money: 0}
{money: 0}*/

补充:
一个应用只能有一个 store,这个时候就会有一个问题,如果有多个 reducer 分别来处理不同的状态,而 createStore 是能接受一个 reducer,这个时候我们就需要 redux 提供的 combineReducers 方法来将多个 reducer 结合成一个 reducer

import {combineReducers} from 'redux';

const reducerFamily=combineReducers({
    reduceSon,
    reduceDaughter,
    reducerFather,
    reducerMother
})
const store = createStore(reducerFamily);

二. 在 React 中使用 redux

如果会 react,那么也一定知道 creact-react-app 这个官方脚手架工具,首先使用 creact-react-app 创建一个项目,然后删除 src 目录下所有文件,接下来就可以愉快的敲代码了。

在 src 下创建三个文件
index.js

import React from 'react'
import ReactDOM from 'react-dom'
import {createStore} from 'redux'
// 引入我们的 reducer 和 action 创建函数
import {reducer, add, subtraction} from './index.redux'
import App from './App'

// 创建 store
const store = createStore(reducer);

//store.subscribe 方法接受的参数是一个函数,// 所以将 ReactDOM.render 方法写在一个函数内
function listen() {
    // 将 store,action 创建函数分别以属性的方式传递给子组件 App
    ReactDOM.render(<App store={store} add={add} subtraction={subtraction}/>,
        document.querySelector('#root'));
}

// 因为刚进入页面没有 dispatch 操作改变 store,// 所以 listen 不会执行,我们需要手动调用一次
listen();

// 重点,改变了 store,页面就会重新渲染,// 可以试试不写这行代码会是怎样的效果
store.subscribe(listen);

App.js

import React from 'react'

export default class App extends React.Component {render() {
        // 从属性中获取 store,action 创建函数
        const {store, add, subtraction} = this.props;
        // 获取 state
        let state = store.getState();
        return <div>
            <h1> 我有 {state.money} 元 </h1>

            {/* 通过 store.dispatch 方法改变 store,从而页面也会改变 */}
            <button onClick={() => {store.dispatch(add())}}>
                捡了一块钱
            </button>

            <button onClick={() => {store.dispatch(subtraction())}}>
                掉了一块钱
            </button>
        </div>
    }
}

index.redux.js

// 定义常量方便维护
const ADD = '+', SUBTRACTION = '-';

// 给初始状态一个默认值:{money: 0}
export function reducer(state = {money: 0}, action) {// 返回一个新的 state 可以使用 es6 提供的 Object.assign()方法,或扩展运算符(此方法需要 babel-preset-state- 3 支持)switch (action.type) {
        case ADD:
            return Object.assign({}, state, {money: state.money + 1});
        case SUBTRACTION:
            return {...state, ...{money: state.money - 1}};
        default:
            return state;
    }
}

//action 创建函数,返回了一个 action
export function add() {return {type: ADD}
}

export function subtraction() {return {type: SUBTRACTION}
}

效果图

这样我们就将 redux 和 react 结合了起来但是这样我们可能会觉得麻烦,因为我们要将 store 和 action 创建函数传给子组件,当我们的 action 比较多时,子组件比较多时,就需要将 store 和大量的 action 创建函数一层层的多次传递下去。这样就会十分麻烦,因此我们就可以使用 react-redux 这个库来帮助我们实现这个麻烦的过程

三.react-redux 的使用

1.Provider

react-redux给我们提供了一个 Provider 组件,我们可以把这个组件写在最外层,这样被 Provider 包裹的所有组件都可以通过 props 来获取 state,无论组个组件藏得多么深。
Provider组件只接受一个属性,那就是store

那么我们 index.js 的代码就变成下面这样了:

import React from 'react'
import ReactDOM from 'react-dom'
import {createStore} from 'redux'
import {Provider} from 'react-redux'
import {reducer} from './index.redux'
import App from './App'

// 创建 store
const store = createStore(reducer);

ReactDOM.render(<Provider store={store}>
        <App/>
    </Provider>,
    document.querySelector('#root'));

2.connect

当然,只有 Provider 组件是不够的,我们还需要 connect 来帮助我们获取 state 和 action,没错,connect就是帮助我们获取 state 和 action 的

那么问题就来了,我们的组件可不是需要项目中所有的 state 和 action,只需要其中的一部分就可以了,所以 connect 会接受两个参数,第一个参数它可以帮我们筛选 state,第二个参数可以帮我们筛选 action。
我们可以把这两个参数写成函数的形式,
参数 1,

function mapStateToProps(state) {
    return {money: state.money}
}

参数 2,

function actionCreators() {
    return {
        subtraction,
        add
    }
}

我们可以发现这两个函数都是返回了一个对象,第一个函数返回了我们需要的 state,第二个函数返回了我们需要的 action 创建函数

那么 app.js 的代码就变成这样了:

import React from 'react'
import {connect} from 'react-redux'
import {add, subtraction} from './index.redux'

class App extends React.Component {render() {
        // 因为 connect 的原因,state 和 action 我们已经可以从属性中获取了
        const {money, add, subtraction} = this.props;

        return <div>
            <h1> 我有 {money} 元 </h1>

            {/* 这个时候不需要我们 dispatch 了 */}
            <button onClick={add}>
                捡了一块钱
            </button>

            <button onClick={subtraction}>
                掉了一块钱
            </button>
        </div>
    }
}

//connect 所需要的参数
// 函数返回的我们需要的状态,我们需要 money,就从 state 中取出 money
// 假如我们还需要 house,就增加一个 house:state.house
function mapStateToProps(state) {
    return {money: state.money}
}

//connect 需要的第二参数
// 返回我们需要的 action 创建函数
function actionCreators() {
    return {
        subtraction,
        add
    }
}

// 上面两个函数返回的都是对象

// 通过 connect 将 state 和 action 创建函数当做属性传递给组件
export default App = connect(mapStateToProps, actionCreators())(App);

如果熟悉 es6 装饰器的语法那就更好了,可以使我们的代码变得更优雅
app.js

import React from 'react'
import {connect} from 'react-redux'
import {add, subtraction} from './index.redux'

@connect(state => ({money: state.money}),
    {
        subtraction,
        add
    })
export default class App extends React.Component {render() {
        // 因为 connect 的原因,state 和 action 我们已经可以从属性中获取了
        const {money, add, subtraction} = this.props;

        return <div>
            <h1> 我有 {money} 元 </h1>

            {/* 这个时候不需要我们 dispatch 了 */}
            <button onClick={add}>
                捡了一块钱
            </button>

            <button onClick={subtraction}>
                掉了一块钱
            </button>
        </div>
    }
}

看到这里再回头看看最开始图片,就能搞清楚 redux 的工作流程究竟是怎样的。

正文完
 0