zoukankan      html  css  js  c++  java
  • React中Redux的有关知识

    Redux的设计思想很简单,就两句话。

    * Web应用是一个状态机,视图与状态是一一对应的
    * 所有的状态,保存在一个对象里面

    Store:

    Store就是保存数据的地方,你可以把它看成一个容器。整个应用只能有一个Store。

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

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

    上面代码中,createStore函数接受里呢一个函数作为参数,返回新生成的Store对象。

    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是什么样,反之亦然。

    Action:

    State的变化,会导致View的变化。但是,用户接触不到State,只能接触到View。所以,State的变化时View导致的。Action就是View发出的通知,表示State应该要发生变化了。

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

    const action = {
        type:'ADD_TODO',
        payloadL:'Learn Redux'
    };

    上面代码中,Action的名称是ADD_TODO,它携带的额信息是字符串Learn Redux。

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


    ActionCreator:

    View要发送多少种消息,就会有多少种Action。如果都手写,会很麻烦。可以定义一个函数来生成Action,这个函数就叫ActionCreator。

    const ADD_TODO = '添加 TODO';
    
    function addToDo(text){
        return {
            type:ADD_ToDo,
            text
        }
    }
    
    const action = addToDo('Learn Redux');

    上面代码中,addToDo函数就是一个ActionCreator。


    store.dispatch():

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

    import { createStore } from 'redux';
    const store = createStore(fn);
    
    store.dispatch({
        type:'ADD_TODO',
        payload:'Learn Redux'
    });

    上面代码中,store.dispatch接受一个Action对象作为参数,将它发送出去。

    结合ActionCreator,这段代码可以改写如下。

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

    Reducer:

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

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

    const reducer = function(state,action){
    //...
    return new_state;
    }

    整个应用的初始状态,可以作为State的默认值。下面是一个实际的例子。

    const defaultState = 0;
    
    const reducer = (state = defaultState,action) = >{
    switch(action.type){
    case "ADD":
    return state += action.payload;
    default:
    return state;
    }
    };
    
    const state = reducer(1,{
    type:'ADD',
    payload:2
    });

    上面代码中,reducer函数收到名为ADD的Action以后,就返回一个新的State,作为加法的计算结果。其他运算的逻辑(比如剪发),也可以根据Action的不同来实现。

    实际应用中,Reducer函数不同像上面这样手动调用,store.dispatch方法会触发reducer的自动执行。为此,Store需要知道Reducer函数,做法就是在生成Store的视乎,将Reducer传入createStore方法。

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

    上面代码中,createStore接受Reducer作为参数,生成一个新的Store。以后每当store.dispatch发送过来一个新的Action,就会自动调用Reducer,得到新的State。

    为什么这个函数叫做Reducer呢?因为它可以作为数组的reducer方法的参数。请看下面的例子,一系列Action对象按照顺序作为一个数组。

    const actions = [
    {type:'ADD',payload:0},
    {type:'ADD',payload:1},
    {type:'ADD',payload:2}
    ];
    
    const total = actions.reducer(reducer,0);//3

    上面代码中,数组actions表示依次有三个Action,分别是假0、加1和加2.数组的reducer方法接受Reducer函数作为参数,就可以直接得到最终的状态3。


    纯函数:

    Reducer函数最重要的特征是,它是一个纯函数。也就是说,只要是同样的输入,必须得到同样的输出。

    纯函数是函数式编程的概念,必须遵守以下一些约束。

    * 不得改写参数

    * 不能调用系统I/O的API

    * 不能调用Date.now()或者Math.random()等不纯的方法,因为每次会得到不一样的结果。

    由于Reducer是纯函数,就可以保证同样的State,必定得到同样的View。但也正因为这一点,Reducer函数里面不能改变State,必须返回一个全新的对象,请参考下面的写法。

    //State是一个对象
    function reducer(state,action){
    return Object.assign({},state,{ thingToChange });
    //或者
    return { ...state,...newState };
    }
    
    //State是一个数组
    function reducer(state,action){
    return [...state,newItem];
    }

    最好把State对象设成只读。你没法改变它,要得到新的State,唯一办法就是生成一个新对象。这样的好处是,任何时候,与某个View对应的State总是一个不变的对象。


    store.subscribe():

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

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

    显然,只要把View的更新函数(对于React项目,就是组件的render方法或setState方法)放入listen,就会实现View的自动渲染。

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

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

    Store的实现:

    上一节介绍了Redux涉及的基础概念,可以发现Store提供了三个方法。

    * store.getState()

    * store.dispatch()

    * store.subscribe()

    import { createStore } from 'redux';
    let { subscribe,dispatch,getState } = createStore(reducer);

    createStore方法还可以接受第二个参数,表示State的最初状态。这通常是服务器给出的。

    let store = createStore(todoApp,window.START_FORM_SERVER)

    上面的代码中,window.START_FORM_SERVER就是整个应用的状态初始值。注意,如果提供了这个参数,他会覆盖Reducer函数的默认初始值。

    下面是createStore方法的一个简单实现,可以了解一下 Store 是怎么生成的。

    const createStore = (reducer) => {
    let state;
    let listeners = [];
    
    const getState = () => state;
    
    const dispatch = (action) => {
    state = reducer(state, action);
    listeners.forEach(listener => listener());
    };
    
    const subscribe = (listener) => {
    listeners.push(listener);
    return () => {
    listeners = listeners.filter(l => l !== listener);
    }
    };
    
    dispatch({});
    
    return { getState, dispatch, subscribe };
    };

    Reducer 的拆分:

    Reducer 函数负责生成 State。由于整个应用只有一个 State 对象,包含所有数据,对于大型应用来说,这个 State 必然十分庞大,导致 Reducer 函数也十分庞大。

    const chatReducer = (state = defaultState, action = {}) => {
    const { type, payload } = action;
    switch (type) {
    case ADD_CHAT:
    return Object.assign({}, state, {
    chatLog: state.chatLog.concat(payload)
    });
    case CHANGE_STATUS:
    return Object.assign({}, state, {
    statusMessage: payload
    });
    case CHANGE_USERNAME:
    return Object.assign({}, state, {
    userName: payload
    });
    default: return state;
    }
    };

    上面代码中,三种 Action 分别改变 State 的三个属性。

    * ADD_CHAT:chatLog属性
    * CHANGE_STATUS:statusMessage属性
    * CHANGE_USERNAME:userName属性

    这三个属性之间没有联系,这提示我们可以把 Reducer 函数拆分。不同的函数负责处理不同属性,最终把它们合并成一个大的 Reducer 即可。

    const chatReducer = (state = defaultState, action = {}) => {
    return {
    chatLog: chatLog(state.chatLog, action),
    statusMessage: statusMessage(state.statusMessage, action),
    userName: userName(state.userName, action)
    }
    };

    上面代码中,Reducer 函数被拆成了三个小函数,每一个负责生成对应的属性。

    这样一拆,Reducer 就易读易写多了。而且,这种拆分与 React 应用的结构相吻合:一个 React 根组件由很多子组件构成。这就是说,子组件与子 Reducer 完全可以对应。

    Redux 提供了一个combineReducers方法,用于 Reducer 的拆分。你只要定义各个子 Reducer 函数,然后用这个方法,将它们合成一个大的 Reducer。

    import { combineReducers } from 'redux';
    
    const chatReducer = combineReducers({
    chatLog,
    statusMessage,
    userName
    })

    export default todoApp;
    上面的代码通过combineReducers方法将三个子 Reducer 合并成一个大的函数。

    这种写法有一个前提,就是 State 的属性名必须与子 Reducer 同名。如果不同名,就要采用下面的写法。

    const reducer = combineReducers({
    a: doSomethingWithA,
    b: processB,
    c: c
    })
    
    // 等同于
    function reducer(state = {}, action) {
    return {
    a: doSomethingWithA(state.a, action),
    b: processB(state.b, action),
    c: c(state.c, action)
    }
    }

    总之,combineReducers()做的就是产生一个整体的 Reducer 函数。该函数根据 State 的 key 去执行相应的子 Reducer,并将返回结果合并成一个大的 State 对象。

    下面是combineReducer的简单实现。

    const combineReducers = reducers => {
    return (state = {}, action) => {
    return Object.keys(reducers).reduce(
    (nextState, key) => {
    nextState[key] = reducers[key](state[key], action);
    return nextState;
    },
    {} 
    );
    };
    };

    你可以把所有子 Reducer 放在一个文件里面,然后统一引入。

    import { combineReducers } from 'redux'
    import * as reducers from './reducers'
    
    const reducer = combineReducers(reducers)

    用自己的话来阐述:

    1. 首先创建store, 2. 然后编写reducer,返回状态,在reducer里面设置默认状态 3. 在组件中引入store,把状态挂载在组件的state中, 4. 在TYPES中创建变量, 5. 在actions中写方法,传入reducer中, 6. 把方法写在ActionCreators中,然后把方法引入actions中, 7. 在actions中利用store.dispatch(action)把action传入到reducer中, 8. 在Reducer中,使用switch来判断action.type, 9. 把actions引入到组件中,调用actions里面的方法, 10. 要想数据改变,需要在组件中更新一下。

  • 相关阅读:
    Atitit 趋势管理之道 attilax著
    Atitit 循环处理的新特性 for...else...
    Atitit 2017年的技术趋势与未来的大技术趋势
    atitit 用什么样的维度看问题.docx 如何了解 看待xxx
    atitit prj mnrs 项目中的几种经理角色.docx
    Atitit IT办公场所以及度假村以及网点以及租房点建设之道 attilax总结
    Atitit 工具选型的因素与方法 attilax总结
    Atitit.团队文化建设影响组织的的一些原理 法则 定理 效应 p826.v4
    Atiitt 管理方面的误区总结 attilax总结
    Atitit 未来趋势把控的书籍 attilax总结 v3
  • 原文地址:https://www.cnblogs.com/llc-url/p/8338373.html
Copyright © 2011-2022 走看看