状态管理(2)- react-redux

在阅读之前,建议先看状态管理1

react-redux 是redux官方推出的,它能够使你的React组件从Redux store 中很方便的读取数据,并且向store中分发actions以此来更新数据。

react-redux 两个重要的成员:

Provider: 它是一个组件,一般包裹在根组件外面,能够使得你的整个app能获取到store中的数据
Connect:这个方法能够是我们的组件和store关联
Connect - 参数说明
有四个参数,两个比较重要:mapStateToProps (state, ownProps) 和 map DispatchToProps (dispatch, ownProps) ; 和两个不常用的mergeProps(stateProps, dispatchProps, ownProps) 和 options

  1. 参数 mapStateToProps:类型 function; 这个函数允许我们将store中的数据作为props绑定到组件上;例如 mapStateToProps(state, ownProps) state: redux 中的store; ownProps: 自己的props; (用于接收action)
  2. mapDispatchToProps(stateProps, dispatchProps, ownProps) 类型: function, 将action作为props绑定到我们自己的函数中,参数:dispatch: 就是store.dispatch() ownProps: 自己的props(用于发送action)

实现一个计数器的例子,来熟悉一下react-redux
初始值为10, 每点一次+, 增加1


image.png

我们将通过两个组件来实现,组件A展示+号按钮, 组件B展示数值, 点击组件A的加号 实现组件B的数据+1

脚手架创建react项目
npx create-react-app redux-demo
安装react-redux
yarn add react-redux
利用redux来构建store
  1. 在src目录下创建 reducer文件夹,在reducer文件夹下添加index.js 文件,构建reducer来响应actions
    reducer接收两个参数,第一个参数是state, 第二个参数是action
// 首先定一个state 的初始值
const initState = {
    count: 10
};
// reducer 需要接受action 然后进行逻辑处理 
// 判断发送过来的action是不是我们需要的
// 如果是我们需要的,就应该return一个新的state
exports.reducer = (state = initState, action) => {
    switch(action.type){
        case 'add_action':
            return {
                count:state.count + 1
            };
        default:
            return state;
    }
}
  1. 在src目录下创建store/index.js 文件,通过createStore方法,把reducer传进来
//  首先需要导入一下createStore 这个方法
import { createStore } from 'redux';
import { reducer } from '../reducer'

// 导入reducer的处理函数 , 构建一个store
// 可以直接将这个store导出
export default createStore(reducer);

上面的两个步骤我们创建了reducer 和 store, 你可能会疑惑action 去哪里了? 一步步来,我们将把action放到对应的组件中

引入Provider组件
  1. 在app.js中引入Provider组件
import { Provider } from 'react-redux
  1. 利用Provider组件将我们整个结构进行包裹,并且传递给store
function App() {
  return (
    <Provider store={store}>
    <div className="App">...</div>
    </Provider>
  );
}
新建组件

在src文件夹下新建page目录,在page目录下新建ComA 和 ComB 文件夹, ComA组件放加号,ComB组件放数值

ComA 组件
  • 在ComA下新建index.js
import React from 'react';
class ComA extends React.Component{
    render(){
        return(
            <button> + </button>
        )
    }
}
export default ComA;
  • 引入connect , Provider只是维护store,connect 才是将组件和store关联起来的
import {connect} from 'react-redux';
  • 在开头我们将来connect的两个比较重要的参数:mapStateToProps (state, ownProps) 和 mapDispatchToProps (dispatch, ownProps) ;
    mapStateToProps 是将store中的数据作为props绑定到组件上
    mapDispatchToProps 是将action作为props绑定到组件上
    这里我们的组件A是+号按钮,是事件发送方,所以不需要第一个参数,只需要实现第二个参数,也就是mapDispatchToProps
    下面让我们实现mapDispatchToProps 它是一个函数,有两个参数
    第一个参数:dispatch: 就是store.dispatch() ownProps: 自己的props(用于发送action)
/**
 *  这个函数要有一个返回值,返回值是一个对象
 * @param {*} dispatch 
 * @returns 
 */
const mapDispatchToProps = (dispatch) => {
    return {
        sendAction:()=>{
            // 利用dispatch 发送一个 action 
            // action对象必须有type属性
            dispatch({
                type:'add_action'
            })
        }
    }
}
  • 将组件和 mapDispatchToProps 用connect绑定,并为按钮添加点击事件, 点击事件就是从mapDispatchToProps return出来的方法
    ComA完整代码:
import React from 'react';
// 导入connect
import {connect} from 'react-redux';

class ComA extends React.Component{
    handleClick = () => {
        console.log(this.props,'ppp')
        // 把sendAction传过来了
        // 这里我们就发送action
        this.props.sendAction();
    }
    render(){
        return(
            <button onClick={this.handleClick}> + </button>
        )
    }
}
/**
 *  这个函数要有一个返回值,返回值是一个对象
 * @param {*} dispatch 
 * @returns 
 */
const mapDispatchToProps = (dispatch) => {
    return {
        sendAction:()=>{
            // 利用dispatch 发送一个 action 
            // action对象必须有type属性
            dispatch({
                type:'add_action'
            })
        }
    }
}

// 组件A是发送方, 所以要实现第二个参数
export default connect(null, mapDispatchToProps)(ComA)
ComB 组件

ComB用于显示数值,它是接受state的,所以只要实现connect中的mapStateToProps , 它store中的数据作为props绑定到组件上, 这里我们定义 mapStateToProps, 让它直接将state返回, 那么组件B中就能直接使用props中的数值

import React from 'react'
// 导入connect
import { connect  } from 'react-redux'
class ComB extends React.Component{
    render(){
        return (
            <div>{this.props.count}</div>
        )
    }
}

/**
 *  接受两个参数
 * @param {*} state 
 */
const mapStateToProps = (state) => {
    return state;
}
export default connect(mapStateToProps)(ComB)
最后将ComA 和 ComB 引入app.js 中

app.js 完整代码

import './App.css';
import store from './store'
// 导入Provider组件,利用这个组件包裹我们的结构,从而能够达到统一维护store的效果
import {Provider} from 'react-redux'
import ComA from './page/ComA'
import ComB from './page/ComB'
// 导入store
function App() {
  return (
    <Provider store={store}>
    <div className="App">
      <ComA />
      <ComB />
    </div>
    </Provider>
  );
}

export default App;

以上就完成了一个简单的计数器

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 215,245评论 6 497
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,749评论 3 391
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 160,960评论 0 350
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,575评论 1 288
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,668评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,670评论 1 294
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,664评论 3 415
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,422评论 0 270
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,864评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,178评论 2 331
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,340评论 1 344
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,015评论 5 340
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,646评论 3 323
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,265评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,494评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,261评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,206评论 2 352

推荐阅读更多精彩内容