Redux实现组合计数器

Redux是一种解决数据共享的方案

import {createStore} from 'redux';
import React from 'react';
import ReactDOM from 'react-dom';
import {connect, createProvider} from 'react-redux'


// data
let allNum = {num :1000}

// 创建reducer, 名字的默认值为
function reducer(state, action) {
    let tmp = {}
    if (action.type == "decrease"){
        allNum.num = allNum.num - action.value;
        tmp = Object.assign({}, state, {num: allNum.num})
        return tmp
    }else if(action.type == "increase"){
        allNum.num = allNum.num + action.value;
        tmp = Object.assign({}, state, {num: allNum.num})
        return tmp
    }else{
        return state
    }
}

// 创建store存储数据(传入处理函数reducer, 核心数据allNum)
let store = createStore(reducer, allNum)
console.log("初始化的数据为",store.getState('num'))

// 添加监听函数
store.subscribe(() => {console.log("监听函数发出:", store.getState())});

// 发出action
let tmp = {};
tmp = store.dispatch({type: "decrease", value: 10})
console.log("---->", tmp);
tmp = store.dispatch({type: "decrease", value: 100})
console.log("---->", tmp);
tmp = store.dispatch({type: "increase", value: 31})
console.log("---->", tmp);
tmp = store.dispatch({type: "increase", value: 123})
console.log("---->", tmp);

class MyComponent extends React.Component {
  render() {return <div>Hello World</div>;}
}

ReactDOM.render(<MyComponent />, document.getElementById("root"));

React和Redux组合使用

  • React组件, 有两个数据集, propsstate
  • props表示外部传入组件的参数(数据由外部传入, 可以被外部更改)
  • state表示组件固有的属性(数据私有, 不可以被外部更改)
  • 我们可以把多个React组件的props交由Redux进行管理, 这样就实现了React组件之间数据的共享

组件如何读写数据

组件通过action发送信号, reducer处理action, story内的值被reducer修改, 由于React组件已经被绑定到story中, 所以story内的数据被修改后, 可以直接同步到React的组件中


小案例: 实现一个组合计数器

  • 单个计数器的数据由组件自身state管理
  • 三个计数器的数据只和由Redux管理
动图演示
  • 实现的源码如下

index.html

<!DOCTYPE html>
<html lang="en">
<head>
    <meta charset="UTF-8">
    <title>react-webpack-demo</title>
</head>
<body>
    <div id="root"></div>
</body>
</html>

index.js

import 'babel-polyfill';
import React from 'react';
import ReactDOM from 'react-dom';
import './index.scss';
import Redux from 'redux';
import { connect, Provider } from 'react-redux';
import { createStore } from 'redux';
import { PropTypes } from 'prop-types';

class ManageCounter extends React.Component {
    constructor(props) {
        super(props);
    }

    render() {
        return ( <div>
            <p className="title">计数器</p> 
            <Counter id = "0" />
            <Counter id = "1" />
            <Counter id = "2" />
            <p className="result"> 组件值的和为: { this.props.sum } </p> 
            </div> )
    }
}


class Counter extends React.Component {
    constructor(props) {
        super(props);
        this.changeSum = this.changeSum.bind(this)
        this.decrease = this.decrease.bind(this)
        this.increase = this.increase.bind(this)
        this.state = { value: 0 };
    }
    changeSum() {
        this.props.dispatch({ type: 'changeSum', payload: { id: this.props.id, value: this.state.value } })
    }
    decrease() {
        let self = this;
        this.setState({ value: this.state.value - 1 }, () => {
            self.changeSum()

        })
    }

    increase() {
        let self = this;
        self.setState({ value: this.state.value + 1 }, () => {
            self.changeSum()
        })
    }

    render() {
        const { value } = this.state;
        let { id } = this.props;
        return ( <div >
            <input type = "button"value = "减1" onClick = { this.decrease }/>  
            <span > { value } < /span><br/ >
            <input type = "button" value = "加1" onClick = { this.increase }/>
            </div> )
    }
}

// 创建reducer
function reducer(state = { number: [0, 0, 0], sum: 0 }, action = {}) {
    if (action.type == 'changeSum') {
        let { id, value } = action.payload
        console.log("id:", id, "value:", value);
        state.number[id] = value
        let tmpSum = 0;
        for (let i = 0; i < state.number.length; i++) {
            tmpSum += state.number[i]
        }
        return Object.assign({}, state, { sum: tmpSum });
    } else {
        return state;
    }
}

const CounterMapStateToProps = (state) => ({

})

const ManageCounterMapStateToProps = (state) => ({
    sum: state.sum
})

const mapDispatchToProps = (dispatch) => ({
    dispatch: dispatch
})


// 创建store
let store = createStore(reducer)
// connect连接
Counter = connect(CounterMapStateToProps, mapDispatchToProps)(Counter)
ManageCounter = connect(ManageCounterMapStateToProps, mapDispatchToProps)(ManageCounter)

ReactDOM.render(
    <Provider store = { store }>
    <ManageCounter />
    </Provider> ,
    document.getElementById('root'));

index.scss

$designWidth: 750;
@function px2rem($px) {
    @return $px*10/$designWidth+rem;
}

#root {
    div {
        p {
            font-size: px2rem(300);
            color: #5EA1F3;
            text-align: center;
        }
        div {
            font-size: px2rem(500);
            display: flex;
            color: #64B587;
            border: 1px solid #F0BB40;
            input {
                flex: 1 1 auto;
                background-color: #64B587;
                font-size: px2rem(200);
                outline: none;
                color:#ffffff;
            }
            span {
                width: 300px;
                flex: 1 1 auto;
                text-align: center;
            }
        }
        .title {
            color: #BDBDBD;
        }
        .result {

            font-size: px2rem(200);
        }
    }
}

小结

redux的设计思想是很简单的, 也有了很成熟的库函数供我们调用, 所以面对一个问题时, 我们考虑的重点是: React组件内哪些数据需要被Redux管理?把重点问题考虑清楚, 问题也就解决了大半!

为便于管理, 相关资源整合到一张独立的帖子,链接如下:
http://www.jianshu.com/p/4f28e1ae08b1

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

推荐阅读更多精彩内容