当一个react项目组件层级愈来愈深,页面愈来愈多的时候,数据在各个组件层级和页面之间传递的需求就会比较多,不少变量也须要作成可全局管理的。在这个时候,redux和react-redux的使用就颇有必要了。它们能帮助咱们很方便的进行项目全局性的数据管理。html
下面,就写一下我本身对 redux 和 React-redux 的学习以及使用的心得,权当是对学习过程的一种记录和分享。react
Action 是把数据从应用(这里之因此不叫 view 是由于这些数据有多是服务器响应,用户输入或其它非 view 的数据 )传到 store 的有效载荷。它是 store 数据的惟一来源。通常来讲你会经过 store.dispatch()
将 action 传到 store。redux
Reducers 指定了应用状态的变化如何响应 actions并发送到 store 的,记住 actions 只是描述了有事情发生了这一事实,并无描述应用如何更新 state。api
store就是把action和reducer联系到一块儿的对象,store本质上是一个状态树,保存了全部对象的状态。任何UI组件均可以直接从store访问特定对象的状态。 在 Redux 中,全部的数据(好比state)被保存在一个store容器中 ,在一个应用程序中只能有一个store对象。当一个store接收到一个action,它将把这个action代理给相关的reducer。reducer是一个纯函数,它能够查看以前的状态,执行一个action而且返回一个新的状态。bash
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
}
复制代码
connect 的做用是链接React组件与 Redux store,它包在咱们的容器组件的外一层,它接收上面 Provider 提供的 store 里面的 state 和 dispatch,传给一个构造函数,返回一个对象,以属性形式传给咱们的容器组件。并发
它共有四个参数mapStateToProps, mapDispatchToProps, mergeProps以及options。app
mapStateToProps 的做用是将store里的state(数据源)绑定到指定组件的props中 mapDispatchToProps 的做用是将store里的action(操做数据的方法)绑定到指定组件的props中 另外两个方法通常状况下使用不到,这里就不作介绍。。ide
那么 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的文件夹目录以下
拿管理用户信息数据的需求来举例
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
})
}
}
复制代码
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
}
}
复制代码
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
})();
复制代码
至此就完成了redux+React-redux的一个简单使用流程