zoukankan      html  css  js  c++  java
  • react中redux的原理以及使用

    学习必备要点:

    1. 首先弄明白,Redux在使用React开发应用时,起到什么作用——状态集中管理
    2. 弄清楚Redux是如何实现状态管理的——store、action、reducer三个概念
    3. 在React中集成Redux:redux + react-redux(多了一个概念——selector)
    4. Redux调试工具:redux devtools
    5. redux相关很好用的插件:redux-saga的相关介绍

    redux结构图

     
    react-redux.png

    其中红色虚线部分为redux的内部集成,不能显示的看到。

    • action:是事件,它本质上是JavaScript的普通对象,它描述的是“发生了什么”。action由type:string和其他构成。
    • reducer是一个监听器,只有它可以改变状态。是一个纯函数,它不能修改state,所以必须是生成一个新的state。在default情况下,必须但会旧的state。
    • store是一个类似数据库的存储(或者可以叫做状态树),需要设计自己的数据结构来在状态树中存储自己的数据。

    安装

    npm i --save redux

    附加包

    npm install --save react-redux
    npm install --save-dev redux-devtools

    一、redux和React-redux的几个重要概念

    1.1 action

    Action 是把数据从应用(这里之所以不叫 view 是因为这些数据有可能是服务器响应,用户输入或其它非 view 的数据 )传到 store 的有效载荷。它是 store 数据的唯一来源。一般来说你会通过 store.dispatch() 将 action 传到 store。

    1.2 reducer

    Reducers 指定了应用状态的变化如何响应 actions并发送到 store 的,记住 actions 只是描述了有事情发生了这一事实,并没有描述应用如何更新 state。

    1.3 store

    store就是把action和reducer联系到一起的对象,store本质上是一个状态树,保存了所有对象的状态。任何UI组件都可以直接从store访问特定对象的状态。
    Redux 中,所有的数据(比如state)被保存在一个store容器中 ,在一个应用程序中只能有一个store对象。当一个store接收到一个action,它将把这个action代理给相关的reducer。reducer是一个纯函数,它可以查看之前的状态,执行一个action并且返回一个新的状态。

    1.4 Provider

    Provider 其实就只是一个外层容器,它的作用就是通过配合 connect 来达到跨层级传递数据。使用时只需将Provider定义为整个项目最外层的组件,并设置好store。那么整个项目都可以直接获取这个store。它的原理其实是通过React中的Context来实现的。它大致的核心代码如下:

    import React, {Component} from 'react'
    import {PropTypes} from 'prop-types'
    
    export default class Provider extends Component {
        getChildContext() {
            return {store: this.props.store}
        }
    
        constructor() {
            super()
    
            this.state = {}
        }
    
        render() {
            return this.props.children
        }
    }
    
    Provider.childContextTypes = {
        store: PropTypes.object
    }
    

    1.5 connect

    connect 的作用是连接React组件与 Redux store,它包在我们的容器组件的外一层,它接收上面 Provider 提供的 store 里面的 state 和 dispatch,传给一个构造函数,返回一个对象,以属性形式传给我们的容器组件。

    它共有四个参数mapStateToProps, mapDispatchToProps, mergeProps以及options。

    mapStateToProps 的作用是将store里的state(数据源)绑定到指定组件的props中
    mapDispatchToProps 的作用是将store里的action(操作数据的方法)绑定到指定组件的props中
    另外两个方法一般情况下使用不到,这里就不做介绍。。

    那么 connect 是怎么将React组件与 Redux store连接起来的呢?其主要逻辑可以总结成以下代码:

    import {Component} from "react";
    import React from "react";
    import {PropTypes} from 'prop-types'
    
    const connect = (mapStateToProps, mapDispatchToProps) => (WrappedComponent => {
        class Connect extends Component {
            constructor() {
                super()
    
                this.state = {}
    
            }
    
            componentWillMount() {
                this.unSubscribe = this.context.store.subscribe(() => {
                    this.setState(mapStateToProps(this.context.store.getState()))
                })
            }
    
            componentWillUnmount() {
                this.unSubscribe()
            }
    
            render() {
                return <WrappedComponent  {...this.state}
                                          {...mapDispatchToProps(this.context.store.dispatch)}/>
            }
        }
    
        Connect.contextTypes = {
            store: PropTypes.object
        }
        return Connect
    })
    
    export default connect

    二、redux和React-redux的使用

    项目中关于redux的文件夹目录如下


     
    redux.png

    拿管理用户信息数据的需求来举例

    第一步,编写操作用户信息的action

    
    import {USER_INFO} from "../constants/actionTypes";
    import store from '../store/store'
    
    export const switchUser = (data) => {
        console.log("switchUser()",data);
        return () => {
            store.dispatch({
                type: USER_INFO,
                ...data
            })
        }
    }
    

    第二步,编写改变用户信息并返回新state的reducer

    import {USER_INFO} from "../constants/actionTypes";
    
    const redUserInfo = (state = {
        userId: 10001,
        userName: '',
        userOpenid: '',
        userPhone: '',
        userRole: 0
    }, action) => {
        if (action === undefined) {
            return state
        }
    
        switch (action.type) {
            case USER_INFO:
                return {
                    ...state,
                    ...action
                }
            default:
                return state
        }
    
    }
    

    第三步,完成store的创建

    import {createStore} from 'redux'
    import reducers from '../reducers/index'
    
    let store = createStore(reducers)
    
    export default store
    

    第四步,获取用户信息

    //配置代码,通过connect将组件和store连接起来
    let mapStateToProps = (state) => ({
        userInfo: {...state.redUserInfo}
    })
    
    let mapDispatchToProps = (dispatch) => ({})
    
    export default connect(mapStateToProps, mapDispatchToProps)(PageClass)
    
    //通过props获取用户信息
    this.props.userInfo
    

    第五步,修改用户信息

    import {switchUser} from '../../redux/actions/userInfo'
    
    switchUser({
        userId: 10001,
        userName: '',
        userOpenid: '',
        userPhone: '',
        userRole: 2
    })();


  • 相关阅读:
    Vue浏览器调试工具VueTools安装以及使用
    克莱姆法则 学习
    IfcFacetedBrep —Example Basin faceted brep
    行列式学习
    matlab矩阵旋转任意角度的函数 imrotate
    matlab双杆系统的支撑反力 学习
    matlab矩阵运算——乘法、除法学习
    matlab求航线图问题 学习
    matlab范德蒙矩阵生成学习
    matlab特殊矩阵生成学习
  • 原文地址:https://www.cnblogs.com/cnblogso/p/12357843.html
Copyright © 2011-2022 走看看