浅谈 react-redux

2 篇文章 0 订阅

一、react-redux 介绍

react-redux 是 reactjs 一个衍生插件,在简单的一些应用中,采用 statepropscontext 等就能解决。一旦在业务逻辑复杂时,系统复杂的情况,还是采用这三兄弟去开发,会使得代码很复杂,臃肿,维护显得捉襟见肘,很吃力。react-redux 的出现显得尤为必要,可以很好的系统业务逻辑和前端 ui 代码给分割开,采用函数式编程,以便于维护,以一种比较轻便的桥梁的引入,使得代码更具有维护性,在复杂的业务逻辑面前,也显得轻便很多。

安装

$ yarn add react-redux
//或者
$ cnpm install --save react-redux

二、react-redux 组件

React-Redux 将所有组件分成两大类:UI 组件(presentational component)和容器组件(container component)。

1.UI 组件

UI 组件特征:

  • 只负责 UI 的呈现,不带有任何业务逻辑
  • 没有状态(即不使用this.state这个变量)
  • 所有数据都由参数(this.props)提供
  • 不使用任何 Redux 的 API

UI 组件示例

const test1 = (value) => {
	return (
		<div>{value}</div>
	);
}

因为 UI组件不含有状态,UI组件 又称为 纯组件,即它跟纯函数一样,纯粹由参数决定它的值。

2.容器组件

容器组件特征:

  • 负责管理数据和业务逻辑,不负责 UI 的呈现
  • 带有内部状态
  • 使用 Redux 的 API

UI组件负责 UI 的呈现,容器组件 负责管理数据和逻辑。

3.UI 组件和容器组件的结合

如果一个组件既有 UI 又有业务逻辑,那么将它拆分成两层结构:外面是一个容器组件,里面包了一个UI 组件。前者负责与外部的通信,将数据传给后者,由后者渲染出视图。

React-Redux 规定,所有的 UI组件都由用户提供容器组件 则是由 React-Redux 自动生成。也就是说,用户负责视觉层,状态管理则是全部交给它。

三、react-redux 方法

react-redux 提供了两个方法: connectProvider

1. connect()

React-Redux 提供 connect 方法,用于从 UI 组件生成容器组件。connect 的意思,就是将这两种组件连起来。使用 connect 方法将一个UI组件内部包裹一个容器组件,UI组件如果想要获取数据,从 props 中去获取

import { connect } from 'react-redux'
const VisibleTodoList = connect()(TodoList);  //TodoList是 UI 组件,VisibleTodoList就是由 React-Redux 通过connect方法自动生成的容器组件

因为没有定义业务逻辑,上面这个容器组件毫无意义,只是 UI 组件的一个单纯的包装层。为了定义业务逻辑,需要给出下面两方面的信息:

  • 输入逻辑:外部的数据(即state对象)如何转换为 UI 组件的参数。
  • 输出逻辑:用户发出的动作如何变为 Action 对象,从 UI 组件传出去。

connect 方法的完整 API 如下:

import { connect } from 'react-redux'
const VisibleTodoList = connect(mapStateToProps,mapDispatchToProps)(TodoList)

connect方法接受两个参数:mapStateToPropsmapDispatchToProps。它们定义了 UI 组件的业务逻辑。前者负责输入逻辑,即将 state 映射到 UI 组件的参数(props),后者负责输出逻辑,即将用户对 UI 组件的操作映射成 Action

(1) mapStateToProps()

mapStateToProps 是一个函数。它的作用就是像它的名字那样,建立一个从(外部的)state 对象到(UI 组件的)props对象的映射关系。

React-Redux中的mapStateToProps()

  • mapStateToProps 是一个函数。它的作用是建立一个从外部 state 对象到 UI 组件的 props 对象的映射关系。执行后返回一个对象,里面的每一个键值对就是一个映射。
  • mapStateToProps 会订阅(绑定) Store,每当 state 更新的时候,就会自动执行,重新计算 UI 组件的参数,从而触发 UI 组件的重新渲染。
  • mapStateToProps 的第一个参数总是 state 对象,还可以使用第二个参数,代表容器组件的 props 对象。使用 ownProps 作为参数后,如果容器组件的参数发生变化,也会引发 UI 组件重新渲染。
  • 如果 connect 方法省略 mapStateToProps 参数,那么UI 组件就不会订阅 Store,就是说 Store 的更新不会引起 UI 组件的更新。
const mapStateToProps = (state,ownProps)=>{
    return {
    	...
    }
}
(2) mapDispatchToProps()

mapDispatchToPropsconnect 函数的第二个参数,用来建立 UI 组件的参数到 store.dispatch 方法的映射。也就是说,它定义了哪些用户的操作应该当作 Action,传给 Store。它可以是一个函数,也可以是一个对象。

React-Redux中的mapDispatchToProps()

  • 如果 mapDispatchToProps 是一个函数,会得到 dispatchownProps(容器组件的props对象)两个参数,应该返回一个对象,该对象的每个键值对都是一个映射,定义了 UI 组件的参数怎样发出 Action
  • 如果 mapDispatchToProps 是一个对象,它的每个键名也是对应 UI 组件的同名参数,键值应该是一个函数,会被当作 Action creator ,返回的 Action 会由 Redux 自动发出。
//mapDispatchToProps是一个函数
const mapDispatchToProps = (dispatch,ownProps) => {
	return {
		onClick: () => {
			dispatch({
				type: 'SET_VISIBILITY_FILTER',
				filter: ownProps.filter
			});
		}
	};
}

//mapDispatchToProps作为函数
const mapDispatchToProps = {
	onClick: (filter) => {
		type: 'SET_VISIBILITY_FILTER',
		filter: filter
	};
}

connect 的作用是把UI组件(无状态组件)和业务逻辑代码的分开,然后通过 connect 再连接到一起,让代码更加清晰和易于维护。这也是 React-Redux 最大的优点。

2. Provider 组件

connect 方法生成容器组件以后,需要让容器组件拿到 state 对象,才能生成 UI 组件的参数。

一种解决方法是将 state 对象作为参数,传入容器组件。但是容器组件可能在很深的层级,一级级将state 传下去就很麻烦。React-Redux 提供 Provider 组件,可以让容器组件拿到 state。在根组件外面包了一层 Provider,App的所有子组件都可以拿到 state 了。它的原理是React组件的 context 属性,store 放在了上下文对象 context 上面。React-Redux自动生成的容器组件的代码,就类似下面这样,然后子组件就可以从 context 拿到 store

import React from 'react';
import ReactDOM from 'react-dom';
import App from './App';
import {Provider} from 'react-redux';

const store = createStore(reducer);

//直接将数据仓库绑定到根组件,内部的子组件都可以使用
ReactDOM.render(
	<Provider store={store}>    
		<App />
	</Provider>, 
	document.getElementById('root')
)

Provider 在根组件外面包了一层,这样一来,App 的所有子组件就默认都可以拿到 state 了。

总结

  • 想要状态管理机制,使用 Redux
  • 如果想要异步交互,使用 Redux-thunk 或者 Redux-saga 中的一个
  • 如果想要简化代码,获取业务逻辑与UI组件分离,那么使用 react-redux

四、react-redux 示例

Test2.js

//[src/pages/Test2.js]
//Test2组件   用React-redux去做
import React from 'react';
import {connect} from 'react-redux';
import {changeTest2A,toGetDataA} from '../store/actionCreators/test2Action';
const Test2 = (props) =>{
    return (
        <div>
            <h3>react-redux</h3>
            {JSON.stringify(props)}
            <button onClick={props.changeTest2}>修改test2数据</button>
        </div>
    );
}

const mapStateToProps = (state,ownProps)=>{
    return {
        test:state.test2.msg,
        value:'hello',
    }
}
const mapDispatchToProps = (dispatch, ownProps) => {
    return {
        changeTest2: () => {
            //分发action
            dispatch(changeTest2A('1111'));
        }
    }
}

//connect第一个参数mapStateToProps是一个函数,函数有一个参数state,代表store中的所有数据。返回的对象,就是UI组件的props的数据
export default connect(mapStateToProps, mapDispatchToProps)(Test2);

test2Action.js

//[src/store/actionCreators/test2Action.js]
//Test2Action 创建器
import {CHANGE_TEST2} from '../actionTypes';
export const changeTest2A = (value) =>{
    return {
        type: CHANGE_TEST2,
        value
    }
}

test2Reducer.js

//[src/store/reducers/test2Reducer.js]
//Test2reducer  业务逻辑处理
import {CHANGE_TEST2} from '../actionTypes';
let test2State = {
    msg: 'test2',
};
export default (state = test2State, action) => {
    if (action.type === CHANGE_TEST2) {
        return {
            ...state,
            msg: action.value
        };
    }
    return state;
}

index.js

//[src/store/reducers/index.js]
//整合所有reducer后导出大的reducer
import {combineReducers} from 'redux';
import test2Reducer from './test2Reducer';
const reducer = combineReducers({
    //需要整合reducer(目前就一个reducer,如果有多个需要整合)
    test2: test2Reducer
})

export default reducer;

actionTypes.js

//[src/store/actionTypes.js]
//action的type类型
export const CHANGE_TEST2 = "CHANGE_TEST2";

index.js

//[src/store/index.js]
//store
import {createStore} from 'redux';
import reducer from './reducers';

//将reducer与store绑定,创建store并返回
const store = createStore(reducer);
export default store;

index.js

//[src/index.js]
import React from 'react';
import ReactDOM from 'react-dom';
import './index.css';
import App from './App';
import * as serviceWorker from './serviceWorker';
import {Provider} from 'react-redux';
import store from './store';

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

serviceWorker.unregister();
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值