Redux

介绍 Redux

Redux是一个数据状态管理插件,搭配 React 特别合适,详细的用法可见Redux官网

使用场景

无论是移动端还是 pc 端,当你使用 React 或者 vue 开发组件化的 SPA 程序时,组件之间共享信息是一个非常大的问题。例如,用户登录之后客户端会存储用户信息(如userid、头像等),而系统的很多个组件都会用到这些信息,例如收藏、点赞、评论等。这些组件在用到用户信息时,难道每次使用都重新获取一遍?———— 自然不是这样。因此每个系统都需要一个管理多组件使用的公共信息的功能,这就是 Redux 的作用。同理,vue 也有相应的工具,即 vuex ,可以自己去 github 上搜索相关资料。

初学者可能通过这几句话无法真实理解它的用意,此时你只需要记住:只要使用 React 开发系统,你绝大部分都需要结合 Redux 来使用,后面的课程我们详细讲解 Redux 在实际项目中的使用,课程结束后,你就会明白其中的道理。

安装

如果单纯使用 Redux 仅仅安装 Redux 即可,执行npm install redux --save,不过在 React 中使用 Redux 肯定会用到 react-redux 这一工具,因此这里一起安装完,执行npm install react-redux --save

基本使用

可以参见./app/redux-demo.js中的例子,如下代码

    // 定义计算规则,即 reducer
    function counter(state = 0, action) {
        switch (action.type) {
            case 'INCREMENT':
                return state + 1
            case 'DECREMENT':
                return state - 1
            default:
                return state
        }
    }

    // 根据计算规则生成 store
    let store = createStore(counter)

    // 定义数据(即 state)变化之后的派发规则
    store.subscribe(() => {
        console.log('current state', store.getState())
    })

    // 触发数据变化
    store.dispatch({type: 'INCREMENT'})
    store.dispatch({type: 'INCREMENT'})
    store.dispatch({type: 'DECREMENT'})

简单几十行代码,就诠释了 Redux 的设计理念,这里简单分析一下:

  • Redux 是一个管理数据的工具,我们创建一个store变量用来管理数据。而这个store不是凭空创建的,创建它的前提是,得设定一个管理规则。以上代码中,我们的管理规则是:数据(即state)默认是 0,传入INCREMENT就加一,传入DECREMENT就减一
  • 创建store用来管理数据,具体的管理形式是什么呢?第一,要通过一个函数来触发数据的变化,即dispatch,触发的时候一定要符合之前定制的规则,否则无效。第二,数据一旦发生变化时,会导致怎样后果,即subscribe中定义的函数会执行。第三,如何取得当前的数据,即store.getState()。这一块,熟悉设计模式的同学不难理解,这就是普通的发布和订阅的设计模式,也是js种惯用的设计模式。
  • 还有一点特别要注意,即在规则函数中,数据变化时要return一个新的值,而不是直接修改原来的值。这一点和之前提到的Immutable.js一样,都是使用了不可变数据这一概念。这种设计方式明确了数据的变化时段,使得数据管理更清晰,复杂度更低。

------------------ 分割线 ------------------

Redux 和 React 集成

这块一开始介绍可能会感觉有点混乱,要做好心理准备。

创建 store

跟上次讲过得简单 demo 一样,首先也需要创建一个store,参见./app/store/configureStore.js的代码。之前的 demo 提到,创建store之前要有规则,这里的第一个参数就是这个规则,后面会详细讲到。

    const store = createStore(rootReducer, initialState,
        // 触发 redux-devtools
        window.devToolsExtension ? window.devToolsExtension() : undefined
    )

第二个参数即初始化的数据,第三个参数可调起 chrome 扩展程序,具体可参见 redux-devtools

创建规则(Reducer)

使用 Redux 时,刚才提到的“规则”被称作reducer(就是一个统一的称呼,不比去纠结),因此这里的数据规则代码都在./app/reducers目录下。

先看userinfo.js的代码,跟上次 demo 中的几乎一样,唯一的区别就是将 const 都写到了./app/constants/userinfo.js中。之所以这样做,是因为这些 const 会在多个文件中使用,因此要抽象出来。

再看index.js的代码,它用combineReducers这个函数对userinfo.js的数据进行了封装,这样做是为了更好的扩展性。试想,一个系统中存储在 Redux 中的数据可能会有很多,我们这里已经有一个userinfo.js处理用户数据,和可能哪天就再加nav.js处理导航数据、加ad.js处理广告数据……

上次的demo中,state就是一个数据,可以进行state + 1state - 1,数据结构非常简单。而现在,数据结构复杂太多,必须分组管理。因此我们需要用state.userinfo来表示用户数据,state.nav表示导航数据,state.ad表示广告数据…… ———— 这就是用combineReducers分装各个 reducer 的作用。

创建 action

上次的 demo 中,最后执行数据变化时store.dispatch({type: 'INCREMENT'}),这里的{type: 'INCREMENT'}是我们手动写上的,而在实际的应用中,我们需要用一些函数将它分装起来,即./app/actions中的文件,虽然此处只有userinfo.js这一个文件。

userinfo.js中,我们把每个业务操作都分装为一个函数,该函数接收data,然后再根据 reducer 的规则对 data 进行分装,最后返回。当然,最后返回的结果肯定还是会交给dispatch来处理,这是后面要说的。

结合到 React

先看./app/index.js,重点注意下面这些代码。这里,创建了store并传递给<Provider>组件,然后让<Provider>组件作为所有组件的根节点。

import { Provider } from 'react-redux'
import configureStore from './store/configureStore'

const store = configureStore()

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

然后看./containers/Hello.jsx,注意下面这些代码。通过下面的封装,就把userinfouserinfoActions当做props传入到Hello中了,即在Hello组件中通过this.props.userinfothis.props.userinfoActions即可获取数据和 actions

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

function mapDispatchToProps(dispatch) {
    return {
        userinfoActions: bindActionCreators(userinfoActions, dispatch)
    }
}

export default connect(
    mapStateToProps,
    mapDispatchToProps
)(Hello)

获取了数据和 actions 该怎么用呢?我们将它们传递给子组件,AB组件负责展示数据,C组件负责触发actions。具体可参见各个组件的源代码。

    render() {
        return (
            <div>
                <p>hello world</p>
                <hr/>
                <A userinfo={this.props.userinfo}/>
                <hr/>
                <B userinfo={this.props.userinfo}/>
                <hr/>
                <C actions={this.props.userinfoActions}/>
            </div>
        )
    }

运行代码之后,就会看到数据变化的效果了。


总结:接下来的实际项目中,也会像上面这样使用 Redux,如果这里还有点不明,接下来的课程也会再详细讲解————当然,还是要尽早搞明白好

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

推荐阅读更多精彩内容